事务的概念

事务是一组操作的执行单元,相对于数据库操作来讲,事务管理的是一组SQL指令,比如增加,修改,删除等,事务的一致性,要求,这个事务内的操作必须全部执行成功,如果在此过程种出现了差错,比如有一条SQL语句没有执行成功,那么这一组操作都将全部回滚

事务特性(ACID)

  • Atomic(原子性):要么都成功,要么都失败

  • Consistent(一致性):数据应该不被破坏

  • Isolate(隔离性):用户间操作不相混淆

  • Durable(持久性):永久保存

Spring提供了两种事务管理方式

编程式事务管理

编写程序式的事务管理可以清楚的定义事务的边界,可以实现细粒度的事务控制,比如你可以通过程序代码来控制你的事务何时开始,何时结束等,与后面介绍的声明式事务管理相比,它可以实现细粒度的事务控制,例如jdbc,hibernate,spring中不提倡使用。

  • JDBC事务控制
con.setAutoCommite(false);  //设置事务手动提交
  • Hibernate中事务控制
session.beginTransaction();  //开启事务

优缺点:

  1. 事务控制精确

  2. 事务代码,与业务逻辑处理代码,耦合在一起!事务代码,不能共用! 重新写事务控制操作!开发效率低,不便于维护!(不想用事务,要改代码!)

声明式事务管理

如果你并不需要细粒度的事务控制,你可以使用声明式事务,在Spring中,你只需要在Spring配置文件中做一些配置,即可将操作纳入到事务管理中,解除了和代码的耦合, 这是对应用代码影响最小的选择,从这一点再次验证了Spring关于AOP的概念。当你不需要事务管理的时候,可以直接从Spring配置文件中移除该设置

特点:

  1. Spring提供的声明式事务管理,用到Aop概念!

  2. 对指定的方法添加事务控制,这里只需要配置即可!

  3. 修改事务控制实现或删除事务控制操作,只需要移除

注意:只能对某个方法应用事务! (因为“切入点表达式”拦截的是方法,控制不了方法内部代码!)所以,Spring声明式事务管理,即为粗粒度的事务控制!

图片1.png

声明式事务管理 – JDBC

XML 配置方式实现

事务控制在Service层

步骤:

  1. 引入jar文件

    Spring 核心

    Spring Aop 切面编程

    Spring-jdbc / Spring-tx / 驱动包、连接池

  2. dao/service

  3. 配置

  • 数据源

  • JdbcTemplate

  • Dao/Service

  • spring声明式事务管理配置(拦截service方法的执行,动态植入事务控制代码!)

  1. 测试
public interface IDeptDao {

    void save();

}
public interface IDeptService {

    void save();

}
import org.springframework.jdbc.core.JdbcTemplate;

public class DeptDao implements IDeptDao {

    // 注入JdbcTemplate对象
    private JdbcTemplate jdbcTemplate;
    public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
        this.jdbcTemplate = jdbcTemplate;
    }

    public void save() {
        jdbcTemplate.update("insert into t_dept(deptName)values('test..')");
    }
}
public class DeptService implements IDeptService {

    // 注入dao
    private IDeptDao deptDao;
    public void setDeptDao(IDeptDao deptDao) {
        this.deptDao = deptDao;
    }

    public void save() {
        deptDao.save();
        int i = 1/0;
        deptDao.save();
    }
}
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class App {

    @Test
    public void testApp() throws Exception {
        ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml",getClass());
        IDeptService deptService = (IDeptService) ac.getBean("deptService");
        deptService.save();
    }
}
<?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:p="http://www.springframework.org/schema/p"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop" 
    xmlns:tx="http://www.springframework.org/schema/tx"
    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
         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. 数据源配置 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver"></property>
        <property name="jdbcUrl" value="jdbc:mysql:///hib_demo"></property>
        <property name="user" value="root"></property>
        <property name="password" value="root"></property>
        <property name="initialPoolSize" value="3"></property>
        <property name="maxPoolSize" value="6"></property>
    </bean>

    <!-- 2. JdbcTemplate配置 ,  注入数据源-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

    <!-- 3. dao实例,注入jdbcTemplate -->
    <bean id="deptDao" class="cn.itcast.a_tx_jdbc.DeptDao">
        <property name="jdbcTemplate" ref="jdbcTemplate"></property>
    </bean>

    <!-- 4. Service实例,注入dao实例 -->
    <bean id="deptService" class="cn.itcast.a_tx_jdbc.DeptService">
        <property name="deptDao" ref="deptDao"></property>
    </bean>

    <!-- 5. Spring声明式事务管理配置 -->

    <!-- 5.1 配置事务管理器类 -->
    <bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

    <!-- 5.2 事务通知配置, 拦截到指定的方法后如何管理事务 -->
    <!-- find*  find开头的方法,是只读的事务 -->
    <!--   *    上面所有的方法都不满足时候,采用的事务控制规则 -->
    <tx:advice id="txAdvice" transaction-manager="txManager">
        <tx:attributes>
            <tx:method name="find*" read-only="true"/>
            <tx:method name="get*" read-only="true"/>
            <tx:method name="*" read-only="false"/>
        </tx:attributes>
    </tx:advice>

    <!-- 5.3 事务Aop配置 = 切入点表达式  + 应用上面的事务通知 -->
    <aop:config>
        <aop:pointcut expression="execution(* cn.itcast.a_tx_jdbc.*Service.*(..))" id="pt"/>
        <aop:advisor advice-ref="txAdvice" pointcut-ref="pt"/>
    </aop:config>

</beans>

注解方式实现

步骤:

  1. 引入aop相关包

  2. 开启

<tx:annotation-driven transaction-manager="txManager"/>
  1. 使用@Transactional 注解在需要添加事务控制的方法上写这个注解

@Transactional写到方法上, 表示当前方法应用事务控制;写到类上, 表示当前类的所有方法都会应用事务;写到父类上, 当执行父类的这个方法时候才应用事务!

事务属性

// 当前方法应用事务
    @Transactional(
            readOnly=false,      // 读写的事务,当修改数据时候用;如果查询就设置为true
            isolation=Isolation.DEFAULT,  // 事务隔离级别
            timeout=-1,         // 事务执行的超时时间, -1 表示不超时
            noRollbackFor=ArithmeticException.class,   // 遇到指定的异常不回滚
            propagation=Propagation.REQUIRES_NEW      // 事务传播行为
    )

事务传播行为:

图片1.png