`
programmedbloke
  • 浏览: 30073 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

spring声明式事务管理

阅读更多

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

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

Xml代码 复制代码
  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>  
<?xml version="1.0" encoding="gb2312"?>
<!--   Spring配置文件的文件头,包含DTD等信息-->
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
 "http://www.springframework.org/dtd/spring-beans.dtd">
<beans>
    <!--定义数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <!--   定义数据库驱动-->
            <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>
        <!--   定义数据库url-->
            <property name="url"><value>jdbc:mysql://localhost:3306/spring</value></property>
        <!--   定义数据库用户名-->
            <property name="username"><value>root</value></property>
        <!--   定义数据库密码-->
            <property name="password"><value>32147</value></property>
    </bean>
    <!--定义一个hibernate的SessionFactory-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
        <!--   定义SessionFactory必须注入DataSource-->
            <property name="dataSource"><ref local="dataSource"/></property>
            <property name="mappingResources">
            <list>
                <!--以下用来列出所有的PO映射文件-->
                <value>Person.hbm.xml</value>
            </list>
            </property>
            <property name="hibernateProperties">
            <props>
<!--此处用来定义hibernate的SessionFactory的属性:
不同数据库连接,启动时选择create,update,create-drop-->
            <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
            <prop key="hibernate.hbm2ddl.auto">update</prop>
        </props>
        </property>
    </bean>
    <!--   定义事务管理器,使用适用于Hibernte的事务管理器-->
<bean id="transactionManager"
         class="org.springframework.orm.hibernate3.HibernateTransactionManager">
            <!--   HibernateTransactionManager   bean需要依赖注入一个SessionFactory bean的引用-->
            <property name="sessionFactory"><ref local="sessionFactory"/></property>
    </bean>
    <!--定义DAO Bean , 作为事务代理的目标-->
    <bean id="personDaoTarget" class="lee.PersonDaoHibernate">
        <!--   为DAO bean注入SessionFactory引用-->
         <property name="sessionFactory"><ref local="sessionFactory"/></property>
    </bean>
    <!--   定义DAO bean的事务代理-->
<bean id="personDao" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
        <!--   为事务代理bean注入事务管理器-->
            <property name="transactionManager"><ref bean="transactionManager"/></property>
        <!--   设置事务属性-->
        <property name="transactionAttributes">
<props>
      <!--   所有以find开头的方法,采用required的事务策略,并且只读-->
                <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop> 
      <!--   其他方法,采用required的事务策略 ->
                 <prop key="*">PROPAGATION_REQUIRED</prop>
             </props> 
         </property>
        <!--   为事务代理bean设置目标bean -->
        <property name="target"> 
            <ref local="personDaoTarget"/> 
        </property> 
    </bean>
</beans>

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

Xml代码 复制代码
  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>  
<!--   定义DAO bean的事务代理-->
<bean id="personDao" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
         <!--   为事务代理bean注入事务管理器-->
        <property name="transactionManager"><ref bean="transactionManager"/></property>
        <!--   设置事务属性-->
        <property name="transactionAttributes"> 
              <props>
                    <!--   所有以find开头的方法,采用required的事务策略,并且只读-->
                    <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop> 
                    <!--   其他方法,采用required的事务策略 ->
                     <prop key="*">PROPAGATION_REQUIRED</prop>
              </props> 
         </property>
         <!--   为事务代理bean设置目标bean -->
        <property name="target"> 
               <!--   采用嵌套bean配置目标bean-->
               <bean class="lee.PersonDaoHibernate">
                  <!--   为DAO bean注入SessionFactory引用-->
                  <property name="sessionFactory"><ref local="sessionFactory"/></property>
               </bean>
        </property> 
</bean>

 

 

 

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

Xml代码 复制代码
  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>  
<?xml version="1.0" encoding="gb2312"?>
<!--   Spring配置文件的文件头,包含DTD等信息-->
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN"
     "http://www.springframework.org/dtd/spring-beans.dtd">
<beans>
    <!--定义数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <!--   定义数据库驱动-->
            <property name="driverClassName"><value>com.mysql.jdbc.Driver</value></property>
        <!--   定义数据库url-->
            <property name="url"><value>jdbc:mysql://localhost:3306/spring</value></property>
        <!--   定义数据库用户名-->
            <property name="username"><value>root</value></property>
        <!--   定义数据库密码-->
            <property name="password"><value>32147</value></property>
    </bean>
    <!--定义一个hibernate的SessionFactory-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
        <!--   定义SessionFactory必须注入DataSource-->
            <property name="dataSource"><ref local="dataSource"/></property>
            <property name="mappingResources">
            <list>
                <!--以下用来列出所有的PO映射文件-->
                <value>Person.hbm.xml</value>
            </list>
            </property>
            <property name="hibernateProperties">
            <props>
<!--此处用来定义hibernate的SessionFactory的属性:
不同数据库连接,启动时选择create,update,create-drop-->
            <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
            <prop key="hibernate.hbm2ddl.auto">update</prop>
        </props>
        </property>
    </bean>
    <!--   定义事务管理器,使用适用于Hibernte的事务管理器-->
<bean id="transactionManager"
         class="org.springframework.orm.hibernate3.HibernateTransactionManager">
            <!--   HibernateTransactionManager   bean需要依赖注入一个SessionFactory bean的引用-->
            <property name="sessionFactory"><ref local="sessionFactory"/></property>
    </bean>
<!-- 配置事务模板,模板bean被设置成abstract bean,保证不会被初始化-->
<bean id="txBase" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"
 lazy-init="true" abstract="true">
<!--   为事务模板注入事务管理器-->
<property name="transactionManager"><ref bean="transactionManager"/></property>
<!--   设置事务属性-->
        <property name="transactionAttributes"> 
<props> 
         <prop key="find*">PROPAGATION_REQUIRED,readOnly</prop>
         <prop key="*">PROPAGATION_REQUIRED</prop>
    </props> 
</property>
</bean>
<!--   实际的事务代理bean-->
    <bean id="personDao" parent="txBase">
<!--   采用嵌套bean配置目标bean -->
<property name="target">
    <bean class="lee.PersonDaoHibernate">
        <property name="sessionFactory"><ref local="sessionFactory"/></property>
     </bean>
</property>
      </bean>
</beans>

 

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

 

三. 下面介绍一种优秀的事务代理配置策略:采用这种配置策略,完全可以避免增量式配置,所有的事务代理由系统自动创建。容器中的目标bean自动消失,避免需要使用嵌套bean来保证目标bean不可被访问。
这种配置方式依赖于Spring提供的bean后处理器,该后处理器用于为每个bean自动创建代理,此处的代理不仅可以是事务代理,也可以是任意的代理,只需要有合适的拦截器即可。这些是AOP框架的概念,笔者在此处不对AOP进行深入介绍。读者只需了解这种事务代理的配置方式即可。
下面是采用BeanNameAutoProxyCreator配置事务代理的配置文件:

 

Xml代码 复制代码
  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"></sp>
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics