`
sgl124764903
  • 浏览: 174581 次
  • 性别: Icon_minigender_1
  • 来自: 邯郸
社区版块
存档分类
最新评论

转: Spring事务管理----声明式:利用TransactionProxyFactoryBean生成事务代理

    博客分类:
  • SSH2
 
阅读更多

转自:http://blog.csdn.net/cpp_lzth/article/details/6551639

通常建议采用声明式事务管理。声明式事务管理的优势非常明显:代码中无需关于关注事务逻辑,让Spring声明式事务管理负责事务逻辑,声明式事务管理无需与具体的事务逻辑耦合,可以方便地在不同事务逻辑之间切换。
声明式事务管理的配置方式,通常有如下三种:
1.使用TransactionProxyFactoryBean为目标bean生成事务代理的配置。此方式是最传统,配置文件最臃肿、难以阅读的方式。
2.采用bean继承的事务代理配置方式,比较简洁,但依然是增量式配置。
3.使用BeanNameAutoProxyCreator,根据bean name自动生成事务代理的方式,这是直接利用Spring的AOP框架配置事务代理的方式,需要对Spring的AOP框架有所理解。但这种方式避免了增量式配置,效果非常不错。
4.DefaultAdvisorAutoProxyCreator:这也是直接利用Spring的AOP框架配置事务代理的方式,效果也非常不多,只是这种配置方式的可读性不如第三种方式。

一. 利用TransactionProxyFactoryBean生成事务代理
采用这种方式的配置时候,配置文件的增加非常快,每个bean有需要两个bean配置,一个目标,另外还需要使用TransactionProxyFactoryBean配置一个代理bean。
这是一种最原始的配置方式,下面是使用TransactionProxyFactoryBean的配置文件:

 

[xhtml] view plaincopy
  1. <?xml version="1.0" encoding="gb2312"?>  
  2. <!--   Spring配置文件的文件头,包含DTD等信息-->  
  3. <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"  
  4.  "http://www.springframework.org/dtd/spring-beans.dtd">  
  5. <beans>  
  6.     <!--定义数据源-->  
  7.     <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">  
  8.         <!--   定义数据库驱动-->  
  9.             <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>  
  10.         <!--   定义数据库url-->  
  11.             <property name="url"><value>jdbc:mysql://localhost:3306/spring</value></property>  
  12.         <!--   定义数据库用户名-->  
  13.             <property name="username"><value>root</value></property>  
  14.         <!--   定义数据库密码-->  
  15.             <property name="password"><value>32147</value></property>  
  16.     </bean>  
  17.     <!--定义一个hibernate的SessionFactory-->  
  18. <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">  
  19.         <!--   定义SessionFactory必须注入DataSource-->  
  20.             <property name="dataSource"><ref local="dataSource"/></property>  
  21.             <property name="mappingResources">  
  22.             <list>  
  23.                 <!--以下用来列出所有的PO映射文件-->  
  24.                 <value>Person.hbm.xml</value>  
  25.             </list>  
  26.             </property>  
  27.             <property name="hibernateProperties">  
  28.             <props>  
  29. <!--此处用来定义hibernate的SessionFactory的属性:  
  30. 不同数据库连接,启动时选择create,update,create-drop-->  
  31.             <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>  
  32.             <prop key="hibernate.hbm2ddl.auto">update</prop>  
  33.         </props>  
  34.         </property>  
  35.     </bean>  
  36.     <!--   定义事务管理器,使用适用于Hibernte的事务管理器-->  
  37. <bean id="transactionManager"  
  38.          class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  39.             <!--   HibernateTransactionManager   bean需要依赖注入一个SessionFactory bean的引用-->  
  40.             <property name="sessionFactory"><ref local="sessionFactory"/></property>  
  41.     </bean>  
  42.     <!--定义DAO Bean , 作为事务代理的目标-->  
  43.     <bean id="personDaoTarget" class="lee.PersonDaoHibernate">  
  44.         <!--   为DAO bean注入SessionFactory引用-->  
  45.          <property name="sessionFactory"><ref local="sessionFactory"/></property>  
  46.     </bean>  
  47.     <!--   定义DAO bean的事务代理-->  
  48. <bean id="personDao" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">  
  49.         <!--   为事务代理bean注入事务管理器-->  
  50.             <property name="transactionManager"><ref bean="transactionManager"/></property>  
  51.         <!--   设置事务属性-->  
  52.         <property name="transactionAttributes">  
  53. <props>  
  54.       <!--   所有以find开头的方法,采用required的事务策略,并且只读-->  
  55.                 <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>   
  56.       <!--   其他方法,采用required的事务策略 ->  
  57.                  <prop key="*">PROPAGATION_REQUIRED</prop>  
  58.              </props>   
  59.          </property>  
  60.         <!--   为事务代理bean设置目标bean -->  
  61.         <property name="target">   
  62.             <ref local="personDaoTarget"/>   
  63.         </property>   
  64.     </bean>  
  65. </beans>  

 

在上面的配置文件中,personDao需要配置两个部分,一个是personDao的目标bean,该目标bean是实际DAO bean,以实际的DAO bean为目标,建立事务代理。一个组件,需要来个bean组成,一个目标bean,一个事务代理。
这种配置方式还有一个坏处:目标bean直接暴露在Spring容器中,可以直接引用,如果目标bean被误引用,将导致业务操作不具备事务性。
为了避免这种现象,可将目标bean配置成嵌套bean,下面是目标bean和事务代理的配置片段:

 

[xhtml] view plaincopy
  1. <!--   定义DAO bean的事务代理-->  
  2. <bean id="personDao" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">  
  3.          <!--   为事务代理bean注入事务管理器-->  
  4.         <property name="transactionManager"><ref bean="transactionManager"/></property>  
  5.         <!--   设置事务属性-->  
  6.         <property name="transactionAttributes">   
  7.               <props>  
  8.                     <!--   所有以find开头的方法,采用required的事务策略,并且只读-->  
  9.                     <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>   
  10.                     <!--   其他方法,采用required的事务策略 ->  
  11.                      <prop key="*">PROPAGATION_REQUIRED</prop>  
  12.               </props>   
  13.          </property>  
  14.          <!--   为事务代理bean设置目标bean -->  
  15.         <property name="target">   
  16.                <!--   采用嵌套bean配置目标bean-->  
  17.                <bean class="lee.PersonDaoHibernate">  
  18.                   <!--   为DAO bean注入SessionFactory引用-->  
  19.                   <property name="sessionFactory"><ref local="sessionFactory"/></property>  
  20.                </bean>  
  21.         </property>   
  22. </bean>  

 

二. 利用继承简化配置
大部分情况下,每个事务代理的事务属性大同小异,事务代理的实现类都是TransactionProxyFactoryBean,事务代理bean都必须注入事务管理器。
对于这种情况,Spring提供了bean与bean之间的继承,可以简化配置。将大部分的通用配置,配置成事务模板,而实际的事务代理bean,则继承事务模板。这种配置方式可以减少部分配置代码,下面是采用继承的配置文件:

s

[xhtml] view plaincopy
  1. <?xml version="1.0" encoding="gb2312"?>  
  2. <!--   Spring配置文件的文件头,包含DTD等信息-->  
  3. <!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"  
  4.      "http://www.springframework.org/dtd/spring-beans.dtd">  
  5. <beans>  
  6.     <!--定义数据源-->  
  7.     <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">  
  8.         <!--   定义数据库驱动-->  
  9.             <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>  
  10.         <!--   定义数据库url-->  
  11.             <property name="url"><value>jdbc:mysql://localhost:3306/spring</value></property>  
  12.         <!--   定义数据库用户名-->  
  13.             <property name="username"><value>root</value></property>  
  14.         <!--   定义数据库密码-->  
  15.             <property name="password"><value>32147</value></property>  
  16.     </bean>  
  17.     <!--定义一个hibernate的SessionFactory-->  
  18. <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">  
  19.         <!--   定义SessionFactory必须注入DataSource-->  
  20.             <property name="dataSource"><ref local="dataSource"/></property>  
  21.             <property name="mappingResources">  
  22.             <list>  
  23.                 <!--以下用来列出所有的PO映射文件-->  
  24.                 <value>Person.hbm.xml</value>  
  25.             </list>  
  26.             </property>  
  27.             <property name="hibernateProperties">  
  28.             <props>  
  29. <!--此处用来定义hibernate的SessionFactory的属性:  
  30. 不同数据库连接,启动时选择create,update,create-drop-->  
  31.             <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>  
  32.             <prop key="hibernate.hbm2ddl.auto">update</prop>  
  33.         </props>  
  34.         </property>  
  35.     </bean>  
  36.     <!--   定义事务管理器,使用适用于Hibernte的事务管理器-->  
  37. <bean id="transactionManager"  
  38.          class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
  39.             <!--   HibernateTransactionManager   bean需要依赖注入一个SessionFactory bean的引用-->  
  40.             <property name="sessionFactory"><ref local="sessionFactory"/></property>  
  41.     </bean>  
  42. <!-- 配置事务模板,模板bean被设置成abstract bean,保证不会被初始化-->  
  43. <bean id="txBase" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"  
  44.  lazy-init="true" abstract="true">  
  45. <!--   为事务模板注入事务管理器-->  
  46. <property name="transactionManager"><ref bean="transactionManager"/></property>  
  47. <!--   设置事务属性-->  
  48.         <property name="transactionAttributes">   
  49. <props>   
  50.          <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>  
  51.          <prop key="*">PROPAGATION_REQUIRED</prop>  
  52.     </props>   
  53. </property>  
  54. </bean>  
  55. <!--   实际的事务代理bean-->  
  56.     <bean id="personDao" parent="txBase">  
  57. <!--   采用嵌套bean配置目标bean -->  
  58. <property name="target">  
  59.     <bean class="lee.PersonDaoHibernate">  
  60.         <property name="sessionFactory"><ref local="sessionFactory"/></property>  
  61.      </bean>  
  62. </property>  
  63.       </bean>  
  64. </beans>  

 

这种配置方式,相比前面直接采用TransactionProxyFactoryBean的事务代理配置方式,可以大大减少配置文件的代码量。每个事务代理的配置都继承事务模板,无需重复指定事务代理的实现类,无需重复指定事务传播属性——当然,如果新的事务代理有额外的事务属性,也可指定自己的事务属性,此时,子bean的属性覆盖父bean的属性。当然每个事务代理bean都必须配置自己的目标bean,这不可避免。
上面的配置可看出,事务代理的配置依然是增量式的,每个事务代理都需要单独配置——虽然增量已经减少,但每个事务代理都需要单独配置。

分享到:
评论

相关推荐

    spring 事务管理例子(TransactionProxyFactoryBean代理机制 和 tx/aop)

    总结来说,本例子通过`TransactionProxyFactoryBean`和`@Transactional`展示了如何在Spring中实现声明式事务管理,这两种方式都利用了AOP来封装事务逻辑,使代码更加整洁,降低了事务管理的复杂性。在实际应用中,...

    Spring事务相关类Uml图

    本文将深入探讨Spring事务管理的相关类及其UML图,帮助理解这些类之间的关系和作用。 首先,Spring提供了两种事务管理方式:编程式事务管理和声明式事务管理。编程式事务管理通过调用`PlatformTransactionManager`...

    spring事务支持

    Spring的事务管理分为编程式和声明式两种方式,这里主要讨论的是声明式事务管理,因为它是Spring最常用且易于维护的事务管理方式。 首先,Spring的事务管理核心接口是`PlatformTransactionManager`,它定义了开始、...

    详细说明spring事务配置的5种方式

    本文将详细介绍Spring事务配置的五种方式,帮助你深入理解如何在Spring应用中管理事务。 1. **基于XML的声明式事务管理** 第一种方式是在每个Bean上使用代理来实现事务管理。首先,配置`DataSource`,通常是`...

    Spring事务配置5种方式

    -- 配置目标对象,即被代理的对象 --&gt;  /&gt; 在 Spring 中,事务管理主要分为编程式事务管理和声明式事务管理。编程式事务管理需要在代码中显式地调用 begin、commit、rollback 等方法来控制事务,这种方式侵入性较...

    Spring实战之使用TransactionProxyFactoryBean实现声明式事务操作示例

    在Spring框架中,声明式事务管理是通过AOP(面向切面编程)来实现的,它允许开发者在不改变业务代码的情况下控制事务的边界。`TransactionProxyFactoryBean` 是Spring提供的一种工具,用于创建带有事务管理功能的...

    spring事务配置详解

    3. 代理机制:这是Spring事务管理的核心,它决定了何时以及如何应用事务策略。代理可以是AOP(面向切面编程)代理,例如通过`TransactionProxyFactoryBean`或基于注解的事务管理。 以下是Spring中声明式事务配置的...

    spring的5中事物配置 介绍spring的5中事物配置

    Spring 框架提供了多种...在选择事务配置方式时,通常建议使用基于注解或XML的声明式事务管理,因为它们更易于维护和扩展。同时,了解每种配置方式的原理和适用场景,可以帮助我们更好地设计和实现应用程序的事务管理。

    Spring整合JDBC实现转账业务-动态代理模式

    在本示例中,我们将深入探讨如何利用Spring框架与JDBC的整合来实现转账业务,并通过动态代理模式来优化事务管理。动态代理模式是Java中一种强大的设计模式,它允许我们在运行时创建对象的代理,以拦截方法调用并执行...

    SpringXMLTransaction

    首先,我们需要理解Spring事务管理的两种主要模式:编程式事务管理和声明式事务管理。编程式事务管理是通过编程的方式来控制事务的开始、提交、回滚等操作,通常使用`PlatformTransactionManager`接口和`...

    spring_tx_aspectj方式源码

    AspectJ是一种强大的面向切面编程(AOP)工具,它可以与Spring框架集成,用于声明式事务管理。 ### XML方式的AspectJ事务管理 1. **配置引入AspectJ支持**:首先,你需要在Spring配置文件中引入AspectJ的依赖,并...

    Spring事务Transaction配置的五种注入方式详解

    这种方式需要在配置文件中定义一个TransactionProxyFactoryBean,用于生成代理对象。 第二种方式:使用TxAdvice来实现事务管理 在这种方式中,使用TxAdvice来实现事务管理。TxAdvice是一个Advice类型的对象,它...

    struts spring集成【下】

    5. **业务层代理类**:在Spring中,我们通常使用`TransactionProxyFactoryBean`来创建具有事务管理能力的代理类。它会基于AOP动态地生成一个代理,当业务方法执行时,如果发生异常,事务会被回滚;如果没有异常,...

    spring代理详解.txt

    在Spring框架中,代理机制主要用于实现AOP(面向切面编程)功能,通过动态创建代理对象来拦截目标方法调用,从而实现如事务管理、日志记录、性能监控等横切关注点。 ### 1. ProxyFactoryBean `ProxyFactoryBean`是...

    spring框架配置bean的高级属性

    这部分定义了一个事务代理Bean,它主要用于拦截特定的方法并进行事务管理。`transactionAttributes`属性指定了哪些方法需要开启新事务或加入现有事务。 #### 总结 通过对上述XML配置文件的详细解析,我们可以看到...

    springMvc经典面试题

    声明式事务主要通过TransactionProxyFactoryBean来实现,它能够在运行时动态地为目标对象生成一个代理,将事务相关的操作封装在代理中,从而简化事务管理代码。 Spring MVC 还支持与SSH(Struts2、Spring、...

    java web 面试

    Spring框架提供了强大的事务管理功能,其核心组件`TransactionProxyFactoryBean`用于生成事务代理,使得事务管理对应用程序而言是透明的。 Spring支持两种主要的事务管理器: 1. **DataSourceTransactionManager**...

    基于SSH三大框架的实验报告.pdf

    - **Spring的事务管理**:通过`TransactionProxyFactoryBean`类配置事务属性,实现声明式事务管理。 - **Spring的面向切面编程(AOP)**:在Spring中,可以使用AOP为业务逻辑的某些方面提供跨业务逻辑的关注点,例如...

Global site tag (gtag.js) - Google Analytics