Spring学习笔记--事务操作

1. 什么是事务

事务是数据库操作最基本单元,指的是逻辑上的一组操作:要么都成功,如果有一个失败则所有操作都失败。
典型示例:银行转账。

事务具有四大特性(ACID):

  • 原子性
  • 一致性
  • 隔离性
  • 持久性

2. 搭建事务操作环境

以银行转账为示例:
(1)创建数据库
(2)创建service,搭建dao,完成对象创建和注入关系:在service里注入dao,在dao里注入JdbcTemplate,在JdbcTemplate里注入DataSource.
配置文件如下:

<?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:aop="http://www.springframework.org/schema/aop"
       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">

    <!--开启组件扫描-->
    <context:component-scan base-package="com.company"></context:component-scan>

   <!--数据库连接池-->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource" destroy-method="close">
        <property name="url" value="jdbc:mysql://127.0.0.1:3306/user_db?serverTimezone=UTC"/>
        <property name="driverClassName" value="com.mysql.cj.jdbc.Driver"/>
        <property name="username" value="root"/>
        <property name="password" value="root"/>
    </bean>

    <!--JdbcTemplate对象-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <!--注入dataSourse-->
        <property name="dataSource" ref="dataSource"></property>
    </bean>

</beans>

对象注入:
UserService:

@Service
public class UserService {

    @Autowired
    private UserDaO userDaO;
}

UserDaoImpl:

@Repository
public class UserDaoImpl {

    @Autowired
    private JdbcTemplate jdbcTemplate;
}

(3)在dao里创建两个方法:多钱和少钱;在service里创建转账的方法

3. 事务处理过程

(1)开启事务

(2)进行业务操作

(3)无异常,提交事务

(4)出现异常,事务回滚,即回到事务操作之前的状态

3.1 Spring事务管理介绍

(1)事务添加到JavaEE三层结构里面的Service层(即业务逻辑层)

(2)在Spring进行事务管理操作

  • 编程式事务管理(一般开发中不使用)
  • 声明事事务管理(使用):基于注解方式(使用),基于xml配置文件方式

(3)在Spring进行声明式事务管理,底层使用AOP原理

(4)Spring事务管理API:提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类

PlatformTransactionManager,JDBC使用对应DataSourceTransactionManager实现类。

3.1.1 基于注解方式实现声明式事务管理

(1)在Spring配置文件中配置事务管理器

    <!--创建事务管理器-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <!--注入数据源-->
        <property name="dataSource" ref="dataSource"></property>
    </bean>

(2)在Spring配置文件中开启事务注解

  • 引入tx名称空间
  • 开启事务注解
    <!--开启事务注解-->
    <tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>

(3)在service类上面(或service类里的方法上面)添加事务注解

@Transactional.这个注解添加到类上面,也可以添加在方法上面。若把注解添加在类上面,则这个类里面的所有方法都添加事务;若把这个注解添加到方法上面,则为这个方法添加事务。

@Service
@Transactional//事务注解
public class UserService {

    @Autowired
    private UserDaO userDaO;

    public void accountMoney(){

        userDaO.reduceMoney();

        userDaO.addMoney();
    }
}
3.1.2 声明式事务管理参数配置

在service类尚面添加注解@Transactional,在这个注解里可以配置事务相关参数

(1) propagation:事务传播行为。多事务方法直接进行调用,该过程中事务是如何进行管理的。有7种事务传播行为。
事务方法:对数据库表进行变化的操作。
7种事务传播行为参数

(2) isolation:事务隔离级别
不考虑隔离性会产生一下三个读问题:脏读,不可重复读,虚(幻)读;

脏读:一个未提交事务读取到了另一个未提交事务里的数据。
不可重复读:一个未提交事务读取到另一个已提交事务修改的数据。
虚读:一个未提交事务读取到另一个已提交事务添加的数据。

通过设置事务隔离性,可解决读问题。

  • READ UNCOMMITED(读未提交):不可解决上述三个读问题
  • READ COMMITED(读已提交):可解决脏读
  • REPEATABLE READ(可重复读):可解决脏读,不可重复读问题
  • SERIALIZABLE(串行化),可解决上述三个读问题
@Service
@Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.REPEATABLE_READ)
public class UserService {

}

(3) timeout:超时时间,事务需要在一定时间内提交,如果不提交则进行回滚。Spring里默认值为-1,设置时以秒单位进行计算

(4) readOnly:是否只读,默认值为false(表示可以查询,也可以添加修改删除),设置readOnly为true,表示只能查询

(5) rollbackFor:回滚,设置出现哪些异常进行事务回滚

(6) noRollbackFor:不回滚,设置出现哪些异常不进行异常回滚

3.2 基于xml配置文件实现声明式事务管理

(1)配置事务管理器

    <!--1 创建事务管理器-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <!--注入数据源-->
        <property name="dataSource" ref="dataSource"></property>
    </bean>

(2)配置通知

    <!--2 配置通知-->
    <tx:advice id="txadvice">
        <!--配置事务参数-->
        <tx:attributes>
            <!--指定在哪种规则的方法上添加事务-->
            <tx:method name="accountMoney" propagation="REQUIRED"/>
        </tx:attributes>
    </tx:advice>

(3)配置切入点和切面

    <!--3 配置切入点和切面-->
    <aop:config>
        <!--配置切入点-->
        <aop:pointcut id="pointcut" expression="execution(* com.company.service.UserService.*(..))"/>
        <!--配置切面-->
        <aop:advisor advice-ref="txadvice" pointcut-ref="pointcut"/>
    </aop:config>
3.3 完全注解开发实现声明式事务管理

(1)创建配置类TxConfig

@Configuration
@ComponentScan(basePackages = "com.company") //开启组件扫描
@EnableTransactionManagement //开启事务
public class TxConfig {

    //创建数据库连接池
    @Bean
    public DruidDataSource getDruidDataSource(){
        DruidDataSource dataSource=new DruidDataSource();
        dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://127.0.0.1:3306/user_db?serverTimezone=UTC");
        dataSource.setUsername("root");
        dataSource.setPassword("root");
        return dataSource;
    }

    //创建JdbcTemplate对象
    @Bean
    public JdbcTemplate getJdbcTemplate(DataSource dataSource){
        JdbcTemplate jdbcTemplate=new JdbcTemplate();
        //注入dataSource
        jdbcTemplate.setDataSource(dataSource);
        return jdbcTemplate;
    }

    //创建事务管理器
    @Bean
    public DataSourceTransactionManager getDataSourceTransactionManager(DataSource dataSource){
        DataSourceTransactionManager transactionManager=new DataSourceTransactionManager();
        transactionManager.setDataSource(dataSource);
        return transactionManager;
    }
}
©️2020 CSDN 皮肤主题: 数字20 设计师:CSDN官方博客 返回首页