Spring学习之Spring的声明式事务管理详解

声明式事务管理

大多数Spring用户选择声明式事务管理的原因是,这个是对应用代码影响最小的选择,因此也最符合 非侵入式 轻量级容器的理念。

  1. Spring声明式事务管理可以在任何环境下使用。只需更改配置文件, 它就可以和JDBC、JDO、Hibernate或其他的事务机制一起工作。
  2. Spring的声明式事务管理可以被应用到任何类(以及那个类的实例)上。
  3. Spring提供了声明式的回滚规则。
  4. Spring允许你通过AOP定制事务行为。(例如,如果需要,你可以在事务回滚中插入定制的行为。 你也可以增加任意的通知,就象事务通知一样。)。
  5. Spring不提供高端应用服务器提供的跨越远程调用的事务上下文传播。如果你需要这些特性,我们推荐你使用EJB。 然而,不要轻易使用这些特性。因为通常我们并不希望事务跨越远程调用。

理解Spring声明式事务管理的实现

Spring的事务管理是通过AOP代理实现的。 其中的事务通知由元数据(目前基于XML或注解)驱动。

代理对象与事务元数据结合产生了一个AOP代理,它使用一个PlatformTransactionManager 接口配合事务拦截器,在方法调用前后实施事务。

从概念上来说,在事务代理上调用方法的工作过程看起来像这样:

例子

1、首先定义事务性的服务接口:

public interface UserService {

    public abstract void addUser(User user);

    public abstract void deleteUser();

    public abstract void updateUser();

    public abstract List queryUser();

}

2、上面服务的实现类:

public class UserServiceImpl implements UserService {

    private UserDaoImpl userDao;

    @Override
    public void addUser(User user) {
        this.userDao.addUser(user);
    }

    @Override
    public void deleteUser() {
        this.userDao.deleteUser();
    }

    @Override
    public void updateUser() {
        this.userDao.updateUser();
    }

    @Override
    public List queryUser() {
        this.userDao.queryUser();
    }
}

现在假定,UserService的方法(queryUser())必须执行在只读事务上下文中,其他的方法(addUser(User user)、deleteUser()和updateUser())必须执行在可读写事务上下文中。

下面开始配置Spring的配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">

<!-- 声明式事务管理 -->
<!-- 1,引入命名空间:
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation=http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd 

    xmlns:aop="http://www.springframework.org/schema/aop"
    xsi:schemaLocation=http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd
    -->

<!-- 2,配置数据源DataSource -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
    <property name="driverClass" value="com.mysql.jdbc.Driver"/>
    <property name="jdbcUrl" value="jdbc:mysql://localhost:3306/hb_dfdc"/>
    <property name="user" value="root"/>
    <property name="password" value="root"></property>
</bean>

<!-- 3,配置JdbcTemplate,如果不用spring的jdbc可以省略 -->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
    <property name="dataSource" ref="dataSource"/>
</bean>

<!-- 4, 配置DAO层-->
<bean id="userDao" class="com.dfdc.spring.declaratx.dao.impl.UserDaoImpl">
    <property name="jdbcTemplate" ref="jdbcTemplate"/><!--如果不用spring的jdbc可以省略-->
</bean>
<!-- 4.1 配置Service层,即要被事务管理的服务对象 -->
<bean id="userService" class="com.dfdc.spring.declaratx.service.impl.UserServiceImpl">
    <property name="userDao" ref="userDao"/>
</bean>

<!-- 5,配置Spring的事务管理器,即PlatformTransactionManager bean-->
<bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSource"/>
</bean>

<!-- 6, 配置通知-->
<tx:advice id="txAdvice" transaction-manager="txManager">
    <tx:attributes>
        <!-- 所有以query开头的方法是只读的 -->
        <tx:method name="query*" read-only="true"/>
        <!-- 其他方法使用默认的事务设置 -->
        <tx:method name="*"/>
    </tx:attributes>
</tx:advice>

<!-- 7, 启用以上的事务通知-->
<aop:config>
    <!-- 运行被定义在UserServiceImpl类下的任意方法 -->
    <aop:pointcut expression="execution(* com.dfdc.spring.declaratx.service.impl.UserServiceImpl.*(..))" id="aopCut"/>
    <!-- 将切入点与通知编织在一起 -->
    <aop:advisor advice-ref="txAdvice" pointcut-ref="aopCut"/>
</aop:config>

<!-- 其他bean -->

</beans>

我们来分析一下上面的配置。我们要把一个服务对象(’userService’ bean)做成事务性的。

我们想施加的事务语义封装在<tx:advice/>定义中。

<tx:advice/>“把所有以 ‘query’ 开头的方法看做执行在只读事务上下文中, 其余的方法执行在默认语义的事务上下文中”。

其中的 ‘transaction-manager’ 属性被设置为一个指向 PlatformTransactionManager bean的名字(这里指 ‘transactionManager’), 该bean将会真正管理事务。

提示

事实上,如果 PlatformTransactionManager bean的名字是 ‘transactionManager’ 的话,你的事务通知(<tx:advice/>)中的 ‘transaction-manager’ 属性可以忽略。否则你则需要像上例那样明确指定。

配置中最后一段是 <aop:config/> 的定义, 它确保由 ‘txAdvice’ bean定义的事务通知在应用中合适的点被执行。

首先我们定义了一个切面,它匹配 UserService 接口定义的所有操作, 我们把该切面叫做 ‘aopCut’。然后我们用一个通知器(advisor)把这个切面与 ‘txAdvice’ 绑定在一起, 表示当 ‘aopCut’ 执行时,’txAdvice’ 定义的通知逻辑将被执行。

<aop:pointcut/> 元素定义是AspectJ的切面表示法。

一个普遍性的需求是让整个服务层成为事务性的。满足该需求的最好方式是让切面表达式匹配服务层的所有操作方法。例如:

<aop:config>
    <aop:pointcut expression="execution(* com.dfdc.spring.declaratx.service.impl.*.*(..))" id="aopCut"/>
    <aop:advisor advice-ref="txAdvice" pointcut-ref="aopCut"/>
</aop:config>

现在,既然我们已经分析了整个配置,你可能会问了,“好吧,但是所有这些配置做了什么?”。

上面的配置将为’userService’ bean创建一个代理对象,这个代理对象被装配了事务通知,所以当它的相应方法被调用时,一个事务将被启动、挂起、被标记为只读,或者其它(根据该方法所配置的事务语义)。我们来看看下面的测试代码,测试一下上面的配置。

ApplicationContext context = new ClassPathXmlApplicationContext("com/dfdc/spring/declaratx/test/context-declaratx.xml");
UserService userService = (UserService) context.getBean("userService");
userService.deleteUser();

结果:

可以看到spring容器创建了一系列的单例bean,方法deleteUser输入“输出用户”字符串。

设置断点调试:

可以看到userService是由Spring的JDK动态代理生成的代理对象。

总结

Spring声明式事务处理的步骤:

  1. 搭建环境,引入tx和context命名空间;
  2. 在spring的配置文件中,先导入dataSource;
  3. 测试dataSource是否配置正确;(可省略)
  4. 导入dao和service层的bean
  5. 测试dao和service是否配置正确(可省略)
  6. 引入事务管理器
  7. 配置通知<tx:advice/>
  8. 启用事务通知<aop:config/>,将切入点和通知器织入
  9. 测试service层的类,看是否是代理对象。

转载声明

如需转载请注明出处:

http://blog.csdn.net/u011726984

时间: 2024-12-15 01:42:50

Spring学习之Spring的声明式事务管理详解的相关文章

spring声明式事务配置详解

spring声明式事务配置详解 君子不器 2013年06月16日 编程世界 5273次阅读 查看评论 理解Spring的声明式事务管理实现 本节的目的是消除与使用声明式事务管理有关的神秘性.简单点儿总是好的,这份参考文档只是告诉你给你的类加上@Transactional注解,在配置文件中添加('<tx:annotation-driven/>')行,然后期望你理解整个过程是怎么工作的.此节讲述Spring的声明式事务管理内部的工作机制,以帮助你在面对事务相关的问题时不至于误入迷途,回朔到上游平静

全面分析 Spring 的编程式事务管理及声明式事务管理--转

开始之前 关于本教程 本教程将深入讲解 Spring 简单而强大的事务管理功能,包括编程式事务和声明式事务.通过对本教程的学习,您将能够理解 Spring 事务管理的本质,并灵活运用之. 先决条件 本教程假定您已经掌握了 Java 基础知识,并对 Spring 有一定了解.您还需要具备基本的事务管理的知识,比如:事务的定义,隔离级别的概念,等等.本文将直接使用这些概念而不做详细解释.另外,您最好掌握数据库的基础知识,虽然这不是必须. 系统需求 要试验这份教程中的工具和示例,硬件配置需求为:至少带

分析 Spring 的编程式事务管理及声明式事务管理(转)

开始之前 关于本教程 本教程将深入讲解 Spring 简单而强大的事务管理功能,包括编程式事务和声明式事务.通过对本教程的学习,您将能够理解 Spring 事务管理的本质,并灵活运用之. 先决条件 本教程假定您已经掌握了 Java 基础知识,并对 Spring 有一定了解.您还需要具备基本的事务管理的知识,比如:事务的定义,隔离级别的概念,等等.本文将直接使用这些概念而不做详细解释.另外,您最好掌握数据库的基础知识,虽然这不是必须. 系统需求 要试验这份教程中的工具和示例,硬件配置需求为:至少带

Spring 的编程式事务管理及声明式事务管理

本文将深入讲解 Spring 简单而强大的事务管理功能,包括编程式事务和声明式事务.通过对本教程的学习,您将能够理解 Spring 事务管理的本质,并灵活运用之. Spring 事务属性分析 事务管理对于企业应用而言至关重要.它保证了用户的每一次操作都是可靠的,即便出现了异常的访问情况,也不至于破坏后台数据的完整性.就像银行的自助取款机,通常都能正常为客户服务,但是也难免遇到操作过程中机器突然出故障的情况,此时,事务就必须确保出故障前对账户的操作不生效,就像用户刚才完全没有使用过取款机一样,以保

spring事物配置,声明式事务管理和基于@Transactional注解的使用

参考来源:http://blog.csdn.net/bao19901210/article/details/41724355 事物管理对于企业应用来说是至关重要的,好使出现异常情况,它也可以保证数据的一致性. spring支持编程式事务管理和声明式事务管理两种方式. 编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager.对于编程式事务管理,spring推荐使用TransactionTemplate. 声明式事务管理建立在A

spring声明式事务管理

Spring 的声明式事务管理在底层是建立在 AOP 的基础之上的.其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务. 声明式事务管理分为两种:1.配置文件   2.注解 1.配置文件(声明式事务管理)用法: 在applicationContext.xml配置文件中配置①事务管理器(事务管理者).②事务参数(事务通知).③AOP配置 如下: applicationContext.xml配置文件代码 1 <!-- 事务管理器(

Spring声明式事务管理与配置介绍

转至:http://java.9sssd.com/javafw/art/1215 [摘要]本文介绍Spring声明式事务管理与配置,包括Spring声明式事务配置的五种方式.事务的传播属性(Propagation).Spring事务的隔离级别(Isolation level)等内容. 一.Spring声明式事务配置的五种方式 前段时间对Spring的事务配置做了比较深入的研究,在此之间对Spring的事务配置虽说也配置过,但是一直没有一个清楚的认识.通过这次的学习发觉Spring的事务配置只要把

Spring声明式事务管理(基于注解方式实现)

----------------------siwuxie095 Spring 声明式事务管理(基于注解方式实现) 以转账为例 1.导入相关 jar 包(共 10 个包) (1)导入核心 jar 包和日志相关的 jar 包 (2)导入 JdbcTemplate 的 jar 包 (3)导入 MySQL 的 JDBC 驱动包 mysql-connector-java 下载链接: https://dev.mysql.com/downloads/connector/j/ (4)导入 AOP 的 jar

spring 声明式事务管理

简单理解事务: 比如你去ATM机取1000块钱,大体有两个步骤:首先输入密码金额,银行卡扣掉1000元钱:然后ATM出1000元钱.这两个步骤必须是要么都执行要么都不执行.如果银行卡扣除了1000块但是ATM出钱失败的话,你将会损失1000元:如果银行卡扣钱失败但是ATM却出了1000块,那么银行将损失1000元.所以,如果一个步骤成功另一个步骤失败对双方都不是好事,如果不管哪一个步骤失败了以后,整个取钱过程都能回滚,也就是完全取消所有操作的话,这对双方都是极好的. 当这两个步骤提交了,执行完毕