spring事务操作及mysql事务原理

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: spring事务操作及mysql事务原理

@[TOC]

1 事务概念

1、什么是事务
(1)事务是数据库操作最基本单元,逻辑上一组操作,要么都成功,如果有一个失败所有操作都失败
(2)典型场景:银行转账
一个人钱变少,一个人钱变多。
2、事务四个特性(ACID)
(1)原子性(Atomicity):操作不可分割,要么都成功,要么都失败
(2)一致性(Consistency):操作之前和操作之后总量是不变的
(3)隔离性(Isolation):多个事务之间不会产生影响,两个事务操作同一个元素,不会产生影响
(4)持久性(Durability):事务提交后,表中数据发生相应的变化

2 事务操作(搭建事务操作环境)

1、创建数据库,添加数据
在这里插入图片描述
2、创建Service,搭建dao,完成对象创建和注入关系
service注入dao,在dao注入jdbcTemplate,在jdbcTemplate注入DataSource
dao层

public interface IUserDao {
   
   
}
AI 代码解读
@Repository
public class UserDaoImpl implements IUserDao{
   
   
    @Autowired
    private JdbcTemplate jdbcTemplate;
}
AI 代码解读

service层

@Service
public class UserService {
   
   
    @Autowired
    private IUserDao userDaoImpl;
}
AI 代码解读

配置文件

    <!-- 开启注解扫描 -->
    <context:component-scan base-package="com.atguigu.spring5"></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/user_db"></property>
        <property name="username" value="root"></property>
        <property name="password" value="yuan159951."></property>
    </bean>

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

3、在dao创建两个方法:多钱和少钱,在service创建转账的方法
dao层

@Override
    public void addMoney() {
   
   
        String sql = "update t_account set money=money+? where username=?";
        jdbcTemplate.update(sql,100,"张三");
    }

    @Override
    public void reduceMoney() {
   
   
        String sql = "update t_account set money=money-? where username=?";
        jdbcTemplate.update(sql,100,"李四");
    }
AI 代码解读

service层

@Autowired
    private IUserDao userDaoImpl;

    public void accountMoney(){
   
   
        //一个少钱
        userDaoImpl.reduceMoney();
        //一个多钱
        userDaoImpl.addMoney();
    }
AI 代码解读

测试

@Test
    public void testAccountMoney(){
   
   
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService = context.getBean("userService", UserService.class);
        userService.accountMoney();
    }
AI 代码解读

4、上面代码,如果正常执行没有问题,但是如果代码执行过程中出现异常,有问题

public void accountMoney(){
   
   
        //一个少钱
        userDaoImpl.reduceMoney();
        //模拟异常
        int i = 10/0;
        //一个多钱
        userDaoImpl.addMoney();
    }
AI 代码解读

(1)上面问题如何解决呢?
使用事务进行解决
(2)事务操作过程

public void accountMoney(){
   
   
        try {
   
   
            //第一步 开启事务
            //业务逻辑
            //一个少钱
            userDaoImpl.reduceMoney();
            //模拟异常
            int i = 10/0;
            //一个多钱
            userDaoImpl.addMoney();
            //第二步 没有异常则提交事务

        }catch (Exception e){
   
   
            //第三步 有异常则回滚

        }
    }
AI 代码解读

3 事务操作(Spring事务管理介绍)

1、事务添加到JavaEE三层结构里面Service层
2、在Spring进行事务管理操作
(1)有两种方式:编程式事务管理声明式事务管理
3、声明式事务管理
(1)基于注解
(2)基于xml配置文件
4、在Spring进行声明式事务管理,底层使用AOP原理
5、Spring事务管理API
(1)提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类
在这里插入图片描述

4 事务操作(注解声明式事务管理)

1、在spring配置文件中配置事务管理器

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

2、在spring配置文件,开启事务注解
(1)在spring配置问价引入名称空间tx

<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"
       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">
AI 代码解读

(2)开启事务注解

<!-- 开启事务注解 -->
    <tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>
AI 代码解读

3、在Service类上面(或者方法上面)添加事务注解
@Transactional,这个注解添加到类上面,也可以添加方法上面,作用范围不同

    @Autowired
    private IUserDao userDaoImpl;

    @Transactional
    public void accountMoney(){
   
   
//        try {
   
   
            //第一步 开启事务
            //业务逻辑
            //一个少钱
            userDaoImpl.reduceMoney();
            //模拟异常
            int i = 10/0;
            //一个多钱
            userDaoImpl.addMoney();
            //第二步 没有异常则提交事务

//        }catch (Exception e){
   
   
//            //第三步 有异常则回滚
//        }
    }
AI 代码解读

5 事务操作(声明式事务管理参数配置)

1、参数:

@Target({
   
   ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Transactional {
   
   
    @AliasFor("transactionManager")
    String value() default "";

    @AliasFor("value")
    String transactionManager() default "";
    //事务的传播行为
    Propagation propagation() default Propagation.REQUIRED;
      //隔离级别
    Isolation isolation() default Isolation.DEFAULT;
    //超时时间
    int timeout() default -1;
    //只读
    boolean readOnly() default false;
    //回滚
    Class<? extends Throwable>[] rollbackFor() default {
   
   };

    String[] rollbackForClassName() default {
   
   };
    //不回滚
    Class<? extends Throwable>[] noRollbackFor() default {
   
   };

    String[] noRollbackForClassName() default {
   
   };
}
AI 代码解读

2、propagation:表示事务传播行为,当一个事务的方法,被另外的一个事务方法调用的时候,这个事务方法该如何进行处理。
事务传播属性,spring定义了7种
在这里插入图片描述
(1)required:如果方法2有异常回滚,则方法1也回滚
在这里插入图片描述
(2)required_new:如果方法A有异常,回滚了,则方法B不回滚,是一个单独的事务
在这里插入图片描述
(3)supports:和required对比,方法B可以不运行在事务中。
在这里插入图片描述
3、isolation:事务隔离级别
(1)事务有特性称为隔离性,多事务操作之间不会产生影响。
(2)不考虑隔离性产生很多问题。有三个问题:脏读、不可重复度、虚(幻)读。
脏读:一个未提交事务读取到另一个未提交事务的数据,另一个事务可以回滚。事务A读取到的是事务B回滚前的数据。
在这里插入图片描述
不可重复读:一个未提交事务读取到另一个提交事务修改数据
在这里插入图片描述
虚读:一个未提交事务读取到另一个已提交事务添加的数据

(3)解决:通过设置事务隔离性
在这里插入图片描述
mysql默认REPEATABLE Read可重复读。
4、timeout:超时时间
(1)事务需要在一定时间内进行提交,如果不提交进行回滚
(2)默认值是-1,设置时间以秒单位进行设置
5、readOnly:是否只读
(1)读:查询操作,写:添加修改删除操作
(2)readOnly默认值:false,可以查询,可以添加修改删除操作
(3)设置readOnly值为true,设置成true后,只能查询
6、rollbackFor:回滚
(1)设置查询那些异常进行事务回滚
7、norollBackFor:不回滚
(1)设置出现那些异常不进行事务回滚

6 事务操作(XML声明式事务管理)

1、在spring配置文件中进行配置
第一步 配置事务管理器
第二步 配置通知
第三部 配置切入点和切面

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

    <!-- 2、配置通知 -->
    <tx:advice id="txAdvice">
        <!-- 配置事务参数 -->
        <tx:attributes>
            <!-- 指定那种规则的方法上添加事务 -->
            <tx:method name="accountMoney"/>
            <!--<tx:method name="account*"/>-->
        </tx:attributes>
    </tx:advice>

    <!--3、配置切入点和切面-->
    <aop:config>
        <!-- 切入点 -->
        <aop:pointcut id="pt" expression="execution(* com.atguigu.spring5.service.*(..))"/>
        <!-- 配置切面 -->
        <aop:advisor advice-ref="txAdvice" pointcut-ref="pt"/>
    </aop:config>
AI 代码解读

2、测试
去除注解

//    @Transactional(propagation = Propagation.REQUIRED,isolation = Isolation.READ_UNCOMMITTED)
    public void accountMoney(){
   
   
AI 代码解读
@Test
    public void testAccountMoney1(){
   
   
        ApplicationContext context = new ClassPathXmlApplicationContext("bean2.xml");
        UserService userService = context.getBean("userService", UserService.class);
        userService.accountMoney();
    }
AI 代码解读

结果:
在这里插入图片描述

7 事务操作(完全注解声明式事务管理)

1、创建配置陪,替代XML配置文件

@Configuration
@ComponentScan(basePackages = "com.atguigu.spring5")//开启注解扫描
@EnableTransactionManagement//开启事务
public class TxConfig {
   
   

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

    //创建jdbcTemplate
    @Bean  //根据类型注入dataSource
    public JdbcTemplate getJdbcTemplate(DruidDataSource druidDataSource){
   
   
        JdbcTemplate jdbcTemplate = new JdbcTemplate();
        jdbcTemplate.setDataSource(druidDataSource);
        return jdbcTemplate;
    }

    //创建事务管理器
    @Bean
    public DataSourceTransactionManager getDataSourceTransactionManager(DruidDataSource druidDataSource){
   
   
        DataSourceTransactionManager dataSourceTransactionManager = new DataSourceTransactionManager();
        dataSourceTransactionManager.setDataSource(druidDataSource);
        return dataSourceTransactionManager;
    }
}
AI 代码解读

2、测试
记得开启@Transactional注解

@Test
    public void testAccountMoney2(){
   
   
        ApplicationContext context = new AnnotationConfigApplicationContext(TxConfig.class);
        UserService userService = context.getBean("userService", UserService.class);
        userService.accountMoney();
    }
AI 代码解读

结果:
在这里插入图片描述

8 事务原理

  • 原子性、一致性、持久性,依靠redo log和undo log日志实现。
  • 隔离性依靠锁和MVCC实现。

    8.1 MVCC

    8.1.1 MVCC基本概念

  • 当前读
    读取的是记录的最新版本,读取时还要保证其他并发事务不能修改当前记录,会对读取的记录进行加锁。对于我们日常的操作,如:
    select ... lock in share mode(共享锁),select ...for update、update、insert、delete(排他锁)都是一种当前读。
  • 快照读
    简单的select(不加锁)就是快照读,快照读,读取的是记录数据的可见版本,有可能是历史数据,不加锁,是非阻塞读。
    Read Committed: 每次select,都生成一个快照读。
    Repeatable Read:开启事务后第一个select语句才是快照读的地方。
    Serializable:快照读会退化为当前读。
  • MVCC
    全称 Multi-version Concurrency ontrol,多版本并发控制。指维护一个数据的多个版本,使得读写操作没有冲突,快照读为MySOL实现
    MVCC提供了一个非阻塞读功能。MVCC的具体实现,还需要依赖于数据库记录中的三个隐式字段undo log日志readView

    8.1.2 MVCC实现原理

  • 记录中的隐藏字段
    DB_TRX_ID:最近修改事务ID,记录插入这条记录或最后一次修改该记录的事务ID。
    DB_ROLL_PTR:回滚指针,指向这条记录的上一个版本,用于配合undo log,指向上一个版本。
    DB_ROW_ID:隐藏主键,如果表结构没有指定主键,将会生成该隐藏字段。
  • undo log
    回滚日志,在insert、update、delete的时候产生的便于数据回滚的日志。
    当insert的时候,产生的undo log日志只在回滚时需要,在事务提交后,可被立即删除。
    而update、delete的时候,产生的undo log日志不仅在回滚时需要,在快照读时也需要,不会立即被删除.
  • undo log 版本链
    不同事务或相同事务对同一条记录进行修改,会导致该记录的undolog生成一条记录版本链表,链表的头部是最新的旧记录,链表尾部是最
    早的旧记录。
  • readview
    ReadView (读视图)是 快照读 SQL执行时MVCC提取数据的依据,记录并维护系统当前活跃的事务(未提交的)id.
    ReadView中包含了四个核心字段:
    m_ids:当前活跃的事务ID集合
    min_trx_id:最小活跃事务ID
    max_trx_id:预分配事务ID,当前最大事务ID+1 (因为事务ID是自增的)ReadView创建者的事务ID
    creator_trx_id:ReadView创建者的事务ID
    版本链数据访问规则如下:
    在这里插入图片描述

    8.1 隔离性原理

    MVCC(隐藏字段、undolog版本链、ReadView)+锁实现的。

    8.2 一致性原理

    redologundolog共同保证的。

    8.3 持久性原理

  • redo log
    重做日志,记录的是事务提交时数据页的物理修改,是用来实现事务的持久性。
    该日志文件由两部分组成:重做日志缓冲(redo log buffer)以及重做日志文件(redo log fle),前者是在内存中,后者在磁盘中。当事务提交之后会把所有修改信息都存到该日志文件中,用于在刷新脏页到磁盘,发生错误时,进行数据恢复使用。
  • 这种机制就是WAL(Write-Ahead Logging)先写日志。

    8.4 原子性原理

  • undo log
    回滚日志,用于记录数据被修改前的信息,作用包含两个:提供滚和 MVCC(多版本并发控制)。
    undo log和redo log记录物理日志不一样,它是逻辑日志。可以认为当delete一条记录时,undo og中会记录一条对应的inset记录,反之亦然,当update一条记录时,它记录一条对应相反的update记录。当执行rollback时,就可以从undo log中的逻辑记录读取到相应的内容并进行回滚。
    Undo log销毁: undo log在事务执行时产生,事务提交时,并不会立即删除undo log,因为这些日志可能还用于MVCC。
    Undo log存储: undo log采用段的方式进行管理和记录,存放在前面介绍的 rolback segment 回滚段中,内部包含1024个undo logsegment。
相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
目录
打赏
0
0
0
0
277
分享
相关文章
Spring中事务失效的场景
因为Spring事务是基于代理来实现的,所以某个加了@Transactional的⽅法只有是被代理对象调⽤时, 那么这个注解才会⽣效 , 如果使用的是被代理对象调用, 那么@Transactional会失效 同时如果某个⽅法是private的,那么@Transactional也会失效,因为底层cglib是基于⽗⼦类来实现 的,⼦类是不能重载⽗类的private⽅法的,所以⽆法很好的利⽤代理,也会导致@Transactianal失效 如果在业务中对异常进行了捕获处理 , 出现异常后Spring框架无法感知到异常, @Transactional也会失效
MySQL实现文档全文搜索,分词匹配多段落重排展示,知识库搜索原理分享
本文介绍了在文档管理系统中实现高效全文搜索的方案。为解决原有ES搜索引擎私有化部署复杂、运维成本高的问题,我们转而使用MySQL实现搜索功能。通过对用户输入预处理、数据库模糊匹配、结果分段与关键字标红等步骤,实现了精准且高效的搜索效果。目前方案适用于中小企业,未来将根据需求优化并可能重新引入专业搜索引擎以提升性能。
微服务——SpringBoot使用归纳——Spring Boot事务配置管理——常见问题总结
本文总结了Spring Boot中使用事务的常见问题,虽然通过`@Transactional`注解可以轻松实现事务管理,但在实际项目中仍有许多潜在坑点。文章详细分析了三个典型问题:1) 异常未被捕获导致事务未回滚,需明确指定`rollbackFor`属性;2) 异常被try-catch“吃掉”,应避免在事务方法中直接处理异常;3) 事务范围与锁范围不一致引发并发问题,建议调整锁策略以覆盖事务范围。这些问题看似简单,但一旦发生,排查难度较大,因此开发时需格外留意。最后,文章提供了课程源代码下载地址,供读者实践参考。
24 0
微服务——SpringBoot使用归纳——Spring Boot事务配置管理——Spring Boot 事务配置
本文介绍了 Spring Boot 中的事务配置与使用方法。首先需要导入 MySQL 依赖,Spring Boot 会自动注入 `DataSourceTransactionManager`,无需额外配置即可通过 `@Transactional` 注解实现事务管理。接着通过创建一个用户插入功能的示例,展示了如何在 Service 层手动抛出异常以测试事务回滚机制。测试结果表明,数据库中未新增记录,证明事务已成功回滚。此过程简单高效,适合日常开发需求。
31 0
微服务——SpringBoot使用归纳——Spring Boot事务配置管理——事务相关
本文介绍Spring Boot事务配置管理,阐述事务在企业应用开发中的重要性。事务确保数据操作可靠,任一异常均可回滚至初始状态,如转账、购票等场景需全流程执行成功才算完成。同时,事务管理在Spring Boot的service层广泛应用,但根据实际需求也可能存在无需事务的情况,例如独立数据插入操作。
17 0
MySQL原理简介—6.简单的生产优化案例
本文介绍了数据库和存储系统的几个主题: 1. **MySQL日志的顺序写和数据文件的随机读指标**:解释了磁盘随机读和顺序写的原理及对数据库性能的影响。 2. **Linux存储系统软件层原理及IO调度优化原理**:解析了Linux存储系统的分层架构,包括VFS、Page Cache、IO调度等,并推荐使用deadline算法优化IO调度。 3. **数据库服务器使用的RAID存储架构**:介绍了RAID技术的基本概念及其如何通过多磁盘阵列提高存储容量和数据冗余性。 4. **数据库Too many connections故障定位**:分析了MySQL连接数限制问题的原因及解决方法。
MySQL底层概述—9.ACID与事务
本文介绍了数据库事务的ACID特性(原子性、一致性、隔离性、持久性),以及事务控制的演进过程,包括排队、排它锁、读写锁和MVCC(多版本并发控制)。文章详细解释了每个特性的含义及其在MySQL中的实现方式,并探讨了事务隔离级别的类型及其实现机制。重点内容包括:ACID特性(原子性、持久性、隔离性和一致性的定义及其实现方式)、事务控制演进(从简单的全局排队到复杂的MVCC,逐步提升并发性能)、MVCC机制(通过undo log多版本链和Read View实现高效并发控制)、事务隔离级别(析了四种隔离级别(读未提交、读已提交、可重复读、可串行化)的特点及适用场景)、隔离级别与锁的关系。
Spring中的事务是如何实现的
1. Spring事务底层是基于数据库事务和AOP机制的 2. ⾸先对于使⽤了@Transactional注解的Bean,Spring会创建⼀个代理对象作为Bean 3. 当调⽤代理对象的⽅法时,会先判断该⽅法上是否加了@Transactional注解 4. 如果加了,那么则利⽤事务管理器创建⼀个数据库连接 5. 并且修改数据库连接的autocommit属性为false,禁⽌此连接的⾃动提交,这是实现Spring事务⾮ 常重要的⼀步 6. 然后执⾏当前⽅法,⽅法中会执⾏sql 7. 执⾏完当前⽅法后,如果没有出现异常就直接提交事务 8. 如果出现了异常,并且这个异常是需要回滚的就会回滚事务
Spring事务失效,常见的情况有哪些?
本文总结了Spring事务失效的7种常见情况,包括未启用事务管理功能、方法非public类型、数据源未配置事务管理器、自身调用问题、异常类型错误、异常被吞以及业务和事务代码不在同一线程中。同时提供了两种快速定位事务相关Bug的方法:通过查看日志(设置为debug模式)或调试代码(在TransactionInterceptor的invoke方法中设置断点)。文章帮助开发者更好地理解和解决Spring事务中的问题。
RDS用多了,你还知道MySQL主从复制底层原理和实现方案吗?
随着数据量增长和业务扩展,单个数据库难以满足需求,需调整为集群模式以实现负载均衡和读写分离。MySQL主从复制是常见的高可用架构,通过binlog日志同步数据,确保主从数据一致性。本文详细介绍MySQL主从复制原理及配置步骤,包括一主二从集群的搭建过程,帮助读者实现稳定可靠的数据库高可用架构。
107 9
RDS用多了,你还知道MySQL主从复制底层原理和实现方案吗?