• 010-spring事务管理


    一、Spring的事务传播行为

      事务是从哪里传播到哪里?

      是从方法A传播至方法B。

    Spring事务类型详解:
      PROPAGATION_REQUIRED--如果当前没有事务,就新建一个事务。如果有,就加入当前事务。这是最常见的选择。
      PROPAGATION_SUPPORTS--如果当前没有事务,就以非事务方式执行。如果有,就使用当前事务。
      PROPAGATION_MANDATORY--如果当前没有事务,就抛出异常。如果有,就使用当前事务。
      PROPAGATION_REQUIRES_NEW--如果当前没有事务,新建事务。如果有,把当前事务挂起。
      PROPAGATION_NOT_SUPPORTED--如果当前没有事务,以非事务方式执行操作,如果有,就把当前事务挂起。
      PROPAGATION_NEVER--如果当前没有事务,就以非事务方式执行。如果当前存在事务,则抛出异常。
      PROPAGATION_NESTED--如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。
    事务的其他功能

      1、事务超时:设置超时时间,超时后回滚

      2、只读事务:只有读取,提高效率

    二、Spring的事务管理实现

    声明式和编程式[注解式]

    2.1、声明式

      在配置文件中进行相关配置

      Spring配置文件中关于事务配置总是由三个组成部分,分别是DataSource、TransactionManager和代理机制这三部分,无论哪种配置方式,一般变化的只是代理机制这部分。 DataSource、 TransactionManager这两部分只是会根据数据访问方式有所变化,比如使用Hibernate进行数据访问时,DataSource实际为 SessionFactory,TransactionManager的实现为HibernateTransactionManager。 下面一起看看三种声明式事务的具体配置:
      公共配置

    <!-- 配置sessionFactory -->
    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="configLocation">
            <value>classpath:config/hibernate.cfg.xml</value>
        </property>
        <property name="packagesToScan">
            <list>
                <value>com.entity</value>
            </list>
        </property>
    </bean>
    
    <!-- 配置事务管理器(声明式的事务) -->
    <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory"></property>
    </bean>
    
    <!-- 配置DAO --> 
    <bean id="userDao" class="com.dao.UserDaoImpl">
        <property name="sessionFactory" ref="sessionFactory"></property>
    </bean>

    第一种,使用tx标签方式

    <!-- 第一种配置事务的方式 ,tx-->
    <tx:advice id="txadvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="add*" propagation="REQUIRED" rollback-for="Exception" />
            <tx:method name="modify*" propagation="REQUIRED" rollback-for="Exception" />
            <tx:method name="del*" propagation="REQUIRED" rollback-for="Exception"/>
            <tx:method name="*" propagation="REQUIRED" read-only="true"/>
        </tx:attributes>
    </tx:advice>
    
    <aop:config>
        <aop:pointcut id="daoMethod" expression="execution(* com.dao.*.*(..))"/>
        <aop:advisor pointcut-ref="daoMethod" advice-ref="txadvice"/>
    </aop:config>

     expression="execution(* com.dao.*.*(..))"
    其中第一个*代表返回值,第二*代表dao下子包,第三个*代表方法名,“(..)”代表方法参数。

    第二种,使用代理方式

    <!-- 第二种配置事务的方式 ,代理-->
    <bean id="transactionProxy"
        class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"    abstract="true">
        <property name="transactionManager" ref="transactionManager"></property>
        <property name="transactionAttributes">
            <props>
                <prop key="add*">PROPAGATION_REQUIRED, -Exception</prop>
                <prop key="modify*">PROPAGATION_REQUIRED, -Exception</prop>
                <prop key="del*">PROPAGATION_REQUIRED, -Exception</prop>
                <prop key="*">PROPAGATION_REQUIRED, readOnly</prop>
            </props>
        </property>
    </bean>
    <bean id="userDao" parent="transactionProxy">
        <property name="target">
            <!-- 用bean代替ref的方式-->
            <bean class="com.dao.UserDaoImpl">
                <property name="sessionFactory" ref="sessionFactory"></property>
            </bean>
        </property>
    </bean>

     将transactionProxy的abstract属性设置为"true",然后将具体的Dao的parent属性设置为"transactionProxy",可以精简代码。

    第三种,使用拦截器

    <!-- 第三种配置事务的方式,拦截器 (不常用)-->
    <bean id="transactionInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor">
        <property name="transactionManager" ref="transactionManager"></property>
        <property name="transactionAttributes">
            <props>
                <prop key="add*">PROPAGATION_REQUIRED, -Exception</prop>
                <prop key="modify*">PROPAGATION_REQUIRED, -Exception</prop>
                <prop key="del*">PROPAGATION_REQUIRED, -Exception</prop>
                <prop key="*">PROPAGATION_REQUIRED, readOnly</prop>
            </props>
        </property>
    </bean>
    <bean id="proxyFactory" class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">
        <property name="interceptorNames">
            <list>
                <value>transactionInterceptor</value>
            </list>
        </property>
        <property name="beanNames">
            <list>
                <value>*Dao</value>
            </list>
        </property>
    </bean>

    2.2、编程式[注解式]

    编程式即采用注解的方式,需要注意的是,使用注解的方式需要在Spring的配置文件中加入一句话:<context:annotation-config />,其作用是开启注解的方式。具体配置如下:

    <!--开启注解方式-->
    <context:annotation-config />
    
    <!-- 配置sessionFactory -->
    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="configLocation">
            <value>classpath:config/hibernate.cfg.xml</value>
        </property>
        <property name="packagesToScan">
            <list>
                <value>com.entity</value>
            </list>
        </property>
    </bean>
    
    <!-- 配置事务管理器 -->
    <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory"></property>
    </bean>
    
    <!-- 第四种配置事务的方式,注解 -->
    <tx:annotation-driven transaction-manager="transactionManager"/>

    注解文件

    package com.dao;
    
    import org.springframework.orm.hibernate3.HibernateTemplate;
    import org.springframework.transaction.annotation.Propagation;
    import org.springframework.transaction.annotation.Transactional;
    
    import com.entity.User;
    
    @Transactional
    public class UserDaoImpl_BAK extends HibernateTemplate {
    
        @Transactional(propagation=Propagation.REQUIRED,rollbackForClassName="Exception")
        public void addUser(User user) throws Exception {
            this.save(user);
        }
    
        @Transactional(propagation=Propagation.REQUIRED,rollbackForClassName="Exception")
        public void modifyUser(User user) {
            this.update(user);
        }
    
        @Transactional(propagation=Propagation.REQUIRED,rollbackForClassName="Exception")
        public void delUser(String username) {
            this.delete(this.load(User.class, username));
        }
    
        @Transactional(readOnly=true)
        public void selectUser() {
    
        }
    
    }

    类头的@Transactional为默认事务配置,如方法没有自己的事务类型,则按默认事务,如有自己的配置,则按自己的配置。

    以上四种配置方式最常用的还是第一、二种,第三种是比较老旧的方式,而注解的方式不太适合比较大的项目,用于简单的小项目还是很好的,其特点就是简单明了。

     

  • 相关阅读:
    利用xslt合并多个xml文件到一个文件
    如果利用网络推广老家的特产水果?
    C#并行编程中的Parallel.Invoke
    Asp.Net MVC实现优酷(youku)Web的上传
    修改用户名后TSF出现"需要本地工作区。工作区 xxx 并未驻留在本计算机上"
    JS浏览器滚轮事件实现横向滚动照片展
    Android实现dialog时候弹出软键盘dialog移位问题
    快速搭建多线程Windows服务解决方案
    Difference between WCF and Web API and WCF REST and Web Service
    WPF应用程序的性能提升(一)
  • 原文地址:https://www.cnblogs.com/bjlhx/p/8035392.html
Copyright © 2020-2023  润新知