Spring_事务操作:Spring声明式事务管理1(注解方式/参数配置)

注解方式  :注解声明式事务管理

Spring_事务操作:搭建事务操作环境及事务场景引入

1.在Spring配置文件配置事务管理器

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

2.在Spring配置文件,开始事务注解

(1)在Spring配置文件引入名称空间  “tx”

(2)开启事务注解

    <!--开启事务注解-->
    <tx:annotation-driven transaction-manager="dataSourceTransactionManager"></tx:annotation-driven>

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

@Transactional   //添加之后表示:此类里面的所有方法都添加了事务
                  //如果添加到方法上面,只表示此方法有这个事务

完整的xml

<?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: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/tx http://www.springframework.org/schema/tx/spring-tx.xsd">
    <!--开启组件扫描-->
    <context:component-scan base-package="com.tinstu.spring"></context:component-scan>
    <!--配置数据库-->
    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"></property>
        <property name="url" value="jdbc:mysql://localhost:3306/bookjdbc?serverTimezone=UTC"></property>
        <property name="username" value="root"></property>
        <property name="password" value="root"></property>
    </bean>
    <!--JdbcTemplate对象-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <!--注入dataSource-->
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!--创建事务管理器-->
    <bean id="dataSourceTransactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <!--注入数据源-->
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <!--开启事务注解-->
    <tx:annotation-driven transaction-manager="dataSourceTransactionManager"></tx:annotation-driven>
</beans>

在BankService.java上面添加@Transactional注解后,再次带有模拟异常执行,

结果:a和b的钱全部不变!

声明式事务管理参数配置

1.在Service类上面添加注解@Transactional,在这个注解里面可以配置事务相关参数

 

2.propagation :事务传播行为

(1)多事务方法直接进行调用,这个过程中事务是如何进行管理的

如:@Transactional(propagation = Propagation.REQUIRED)

事务的传播行为可以由传播属性指定,Spring定义了7种类型传播行为

常用前两种:

3.ioslatiom: 事务隔离级别

(1)事务有特性称为隔离性:多事务操作之间不会产生影响.不考虑隔离性产生很多问题.

(2)有三个读问题:脏读,不可重复读,虚(幻)读.

  • 脏读:一个未提交事务读取到另一个未提交事务的数据

如:a事务修改小米的余额(100-200),b事务修改小米余额时,读到余额为200,然后进行修改,事务a由于某种原因回滚,则b读到的200余额为脏数据.

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

戏说一下事务中的脏读,幻读,和不可重复读

脏读:事务A在执行的过程修改了数据库中的数据,这时事务B插了进来,读取到了事务A修改之后的数据并且提交了上去,这时事务A由于某些原因进行了回滚,那么事务B读取的就是事务A的脏数据

相当与你在外面吃烧烤,然后有一块肉掉了,恰好掉到了一个外出觅食的蚂蚁面前,蚂蚁立即掉头回去报告,这是你把掉的肉捡了起来,当蚂蚁带他的兄弟们回来的时候,发现没有肉,这时候蚂蚁带给蚁群的就是脏数据。

不可重复读:事务A读取了一组数据,在执行逻辑的时候,事务B修改了这条数据,当事务A再次读取时候,发现数据前后不一致了,这就是不可重复读。

相当与你在外面吃烧烤,掉了一块肉,恰好掉到了一个外出觅食的蚂蚁面前,蚂蚁回去报告,这是你把掉的肉捡了起来,却意外的碰掉了桌子上的韭菜,这是你叹了一口气,说不想捡了,当蚂蚁带他的兄弟们回来的时候,发现这个地方不是肉,而是韭菜,这时候蚂蚁带给蚁群的就是不可重复读数据

幻读:事务A根据条件读取了数据,这时事务B插了进来,修改了数据库里的数据,导致事务A在做第二次查询的时候发现第二次读取的数据比第一次要多,这就是幻读

相当与你在外面吃烧烤,掉了一块肉,恰好掉到了一个外出觅食的蚂蚁面前,蚂蚁回去报告,你不想捡了,这时你又掉了一块,你叹了一口气也没捡,当蚂蚁带他的兄弟们回来的时候,发现这个不是一块肉,而是两块,这时候蚂蚁带给蚁群的就是幻读数据

(3)通过设置事务隔离级别,解决读问题  默认:REPEATABLE_READ

@Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.REPEATABLE_READ)

4.timeout :超时时间

(1)事务需要在一定时间内进行提交,如果不提交进行回滚

(2)默认值是-1,设置时间以秒进行计算

@Transactional(timeout = -1,)

5.readOnly :是否只读

(1)读:查询操作,写:添加修改删除操作

(2)readOnly默认值false,表示可以查询,可以添加修改删除操作.

(3)设置readOnly值是true,设置成TRUE之后,只能查询.

6.rollbackFor:回滚

(1)设置查询哪些异常进行事务回滚

7.norollbackFor:不回滚

(1)设置查询哪些异常不进行事务回滚

阅读剩余
THE END