• Spring第八篇【XML、注解实现事务控制】


    前言

    本博文主要讲解Spring的事务控制,如何使用Spring来对程序进行事务控制….

    一般地,我们事务控制都是在service层做的。。为什么是在service层而不是在dao层呢??有没有这样的疑问…

    service层是业务逻辑层,service的方法一旦执行成功,那么说明该功能没有出错

    一个service方法可能要调用dao层的多个方法…如果在dao层做事务控制的话,一个dao方法出错了,仅仅把事务回滚到当前dao的功能,这样是不合适的。如果没有出错,调用完dao方法就commit了事务,这也是不合适的。


    事务控制概述

    事务控制分为两种:

    • 编程式事务控制
    • 声明式事务控制

    编程式事务控制

    自己手动控制事务,就叫做编程式事务控制。

    • Jdbc代码:
      • Conn.setAutoCommite(false); // 设置手动控制事务
    • Hibernate代码:
      • Session.beginTransaction(); // 开启一个事务
    • 【细粒度的事务控制: 可以对指定的方法、指定的方法的某几行添加事务控制】
    • (比较灵活,但开发起来比较繁琐: 每次都要开启、提交、回滚.)

    声明式事务控制

    Spring提供对事务的控制管理就叫做声明式事务控制

    Spring提供了对事务控制的实现。

    • 如果用户想要使用Spring的事务控制,只需要配置就行了
    • 当不用Spring事务的时候,直接移除就行了。
    • Spring的事务控制是基于AOP实现的。因此它的耦合度是非常低的。
    • 【粗粒度的事务控制: 只能给整个方法应用事务,不可以对方法的某几行应用事务。
      • (因为aop拦截的是方法。)

    Spring给我们提供了事务的管理器类,事务管理器类又分为两种,因为JDBC的事务和Hibernate的事务是不一样的

    • Spring声明式事务管理器类:
      • Jdbc技术:DataSourceTransactionManager
      • Hibernate技术:HibernateTransactionManager

    声明式事务控制

    我们基于Spring的JDBC来做例子吧

    引入相关jar包

    • AOP相关的jar包【因为Spring的声明式事务控制是基于AOP的,那么就需要引入AOP的jar包。】
    • 引入tx名称空间
    • 引入AOP名称空间
    • 引入jdbcjar包【jdbc.jar包和tx.jar包】

    搭建配置环境

    • 编写一个接口
    public interface IUser {
        void save();
    }
    
    • UserDao实现类,使用JdbcTemplate对数据库进行操作!
    
    @Repository
    public class UserDao implements IUser {
    
        //使用Spring的自动装配
        @Autowired
        private JdbcTemplate template;
    
        @Override
        public void save() {
            String sql = "insert into user(name,password) values('zhong','222')";
            template.update(sql);
        }
    
    }
    • userService
    
    @Service
    public class UserService {
    
        @Autowired
        private UserDao userDao;
        public void save() {
    
            userDao.save();
        }
    }
    • bean.xml配置:配置数据库连接池、jdbcTemplate对象、扫描注解
    
    <?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:context="http://www.springframework.org/schema/context"
           xmlns:c="http://www.springframework.org/schema/c"
           xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
    
    
        <!--数据连接池配置-->
        <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
            <property name="driverClass" value="com.mysql.jdbc.Driver"></property>
            <property name="jdbcUrl" value="jdbc:mysql:///zhongfucheng"></property>
            <property name="user" value="root"></property>
            <property name="password" value="root"></property>
            <property name="initialPoolSize" value="3"></property>
            <property name="maxPoolSize" value="10"></property>
            <property name="maxStatements" value="100"></property>
            <property name="acquireIncrement" value="2"></property>
        </bean>
    
        <!--扫描注解-->
        <context:component-scan base-package="bb"/>
    
        <!-- 2. 创建JdbcTemplate对象 -->
        <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
            <property name="dataSource" ref="dataSource"></property>
        </bean>
    
    </beans>

    前面搭建环境的的时候,是没有任何的事务控制的。

    也就是说,当我在service中调用两次userDao.save(),即时在中途中有异常抛出,还是可以在数据库插入一条记录的

    • Service代码:
    
    @Service
    public class UserService {
    
        @Autowired
        private UserDao userDao;
        public void save() {
    
            userDao.save();
    
            int i = 1 / 0;
            userDao.save();
        }
    }
    
    • 测试代码:
    
    public class Test2 {
    
        @Test
        public void test33() {
            ApplicationContext ac = new ClassPathXmlApplicationContext("bb/bean.xml");
    
            UserService userService = (UserService) ac.getBean("userService");
            userService.save();
        }
    }
    

    这里写图片描述


    XML方式实现声明式事务控制

    首先,我们要配置事务的管理器类:因为JDBC和Hibernate的事务控制是不同的。

        <!--1.配置事务的管理器类:JDBC-->
        <bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    
            <!--引用数据库连接池-->
            <property name="dataSource" ref="dataSource"/>
        </bean>
    

    再而,配置事务管理器类如何管理事务

    
        <!--2.配置如何管理事务-->
        <tx:advice id="txAdvice" transaction-manager="txManage">
    
            <!--配置事务的属性-->
            <tx:attributes>
                <!--所有的方法,并不是只读-->
                <tx:method name="*" read-only="false"/>
            </tx:attributes>
        </tx:advice>
    

    最后,配置拦截哪些方法,

    
        <!--3.配置拦截哪些方法+事务的属性-->
        <aop:config>
            <aop:pointcut id="pt" expression="execution(* bb.UserService.*(..) )"/>
            <aop:advisor advice-ref="txAdvice" pointcut-ref="pt"></aop:advisor>
        </aop:config>

    配置完成之后,service中的方法都应该被Spring的声明式事务控制了。因此我们再次测试一下:

    
        @Test
        public void test33() {
            ApplicationContext ac = new ClassPathXmlApplicationContext("bb/bean.xml");
    
            UserService userService = (UserService) ac.getBean("userService");
            userService.save();
        }

    这里写图片描述


    使用注解的方法实现事务控制

    当然了,有的人可能觉得到XML文件上配置太多东西了。Spring也提供了使用注解的方式来实现对事务控制

    第一步和XML的是一样的,必须配置事务管理器类:

    
        <!--1.配置事务的管理器类:JDBC-->
        <bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    
            <!--引用数据库连接池-->
            <property name="dataSource" ref="dataSource"/>
        </bean>
    
    

    第二步:开启以注解的方式来实现事务控制

    
        <!--开启以注解的方式实现事务控制-->
        <tx:annotation-driven transaction-manager="txManage"/>
    

    最后,想要控制哪个方法事务,在其前面添加@Transactional这个注解就行了!如果想要控制整个类的事务,那么在类上面添加就行了。

    
        @Transactional
        public void save() {
    
            userDao.save();
    
            int i = 1 / 0;
            userDao.save();
        }

    这里写图片描述


    事务属性

    其实我们在XML配置管理器类如何管理事务,就是在指定事务的属性!我们来看一下事务的属性有什么:

    这里写图片描述

    对于事务的隔离级别,不清楚的朋友可参考我之前的博文:http://blog.csdn.net/hon_3y/article/details/53760782

    事务传播行为:

    看了上面的事务属性,没有接触过的其实就这么一个:propagation = Propagation.REQUIRED事务的传播行为。

    事务传播行为的属性有以下这么多个,常用的就只有两个:

    • Propagation.REQUIRED【如果当前方法已经有事务了,加入当前方法事务
    • Propagation.REQUIRED_NEW【如果当前方法有事务了,当前方法事务会挂起。始终开启一个新的事务,直到新的事务执行完、当前方法的事务才开始】

    这里写图片描述

    当事务传播行为是Propagation.REQUIRED

    • 现在有一个日志类,它的事务传播行为是Propagation.REQUIRED
    
        Class Log{
                Propagation.REQUIRED  
                insertLog();  
        }
    • 现在,我要在保存之前记录日志
    
        Propagation.REQUIRED
        Void  saveDept(){
            insertLog();   
            saveDept();
        }

    saveDept()本身就存在着一个事务,当调用insertLog()的时候,insertLog()的事务会加入到saveDept()事务中

    也就是说,saveDept()方法内始终是一个事务,如果在途中出现了异常,那么insertLog()的数据是会被回滚的【因为在同一事务内】

    
        Void  saveDept(){
            insertLog();    // 加入当前事务
            .. 异常, 会回滚
            saveDept();
        }

    当事务传播行为是Propagation.REQUIRED_NEW

    • 现在有一个日志类,它的事务传播行为是Propagation.REQUIRED_NEW
    
        Class Log{
                Propagation.REQUIRED  
                insertLog();  
        }
    • 现在,我要在保存之前记录日志
    
        Propagation.REQUIRED
        Void  saveDept(){
            insertLog();   
            saveDept();
        }

    当执行到saveDept()中的insertLog()方法时,insertLog()方法发现 saveDept()已经存在事务了,insertLog()会独自新开一个事务,直到事务关闭之后,再执行下面的方法

    如果在中途中抛出了异常,insertLog()是不会回滚的,因为它的事务是自己的,已经提交了

    
        Void  saveDept(){
            insertLog();    // 始终开启事务
            .. 异常, 日志不会回滚
            saveDept();
        }

  • 相关阅读:
    mac 快捷键
    mac 配置nginx 虚拟域名(转载)
    StringUtils中 isNotEmpty 和isNotBlank的区别【java字符串判空】
    软件常用版本英文snapshot和ga
    IF条件控制
    函数与方法
    数据类型
    函数 FUNCTION
    集合 SET
    字典 DICTIONARY
  • 原文地址:https://www.cnblogs.com/zhong-fucheng/p/7202948.html
Copyright © 2020-2023  润新知