声明式事务---Spring源码从入门 到精通 (二十五 )

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 声明式事务---Spring源码从入门 到精通 (二十五 )

上篇文章aop源码运行总结:

AOP总结目录--- Spring源码从入门到精通(二十四)


这篇文章主要介绍事务的作用 :


事务可以在运行的业务发生异常时,保证这里面对数据库的操作都进行回滚,保证原子性,要么全部成功,要么全部失败,不会出现张三扣款5元成功,李四未收到5元转账。


先贴上项目目录:

image.png

一、事务环境搭建


先导入我们需要的数据源,数据库,springjdbc,数据源c3p0和 数据库 mysql我们之前已经导入过maven包,接下来导入springjdbc的maven:

 <!--jdbc-->
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-jdbc</artifactId>
        <version>4.3.12.RELEASE</version>
    </dependency>

接下来创建新的配置类,里面加DataSource连接数据库,在创建JdbcTemplate对象加载入ioc容器。写一个dao和service,dao里面@Autowired注入jdbcTemplate连接数据库,sql操作数据。

/**
 * 事务配置类
 * 环境搭建:数据源,数据库、spring-jdbc
 * 配置数据源:JdbcTemplate操作数据(spring提供的操作数据工具)
 *
 * @author keying
 */
@Configuration
@ComponentScan("com.alibaba.tx")
public class TxConfig {
    @Bean
    public DataSource dataSource() throws Exception {
        ComboPooledDataSource dataSource = new ComboPooledDataSource();
        dataSource.setDriverClass("com.mysql.jdbc.Driver");
        dataSource.setJdbcUrl("jdbc:mysql://127.0.0.1:3306/testmac");
        dataSource.setUser("root");
        dataSource.setPassword("root");
        return dataSource;
    }
    /**
     * spring对@Configuration有特殊处理,给容器中加组件的方法,多次调用也只是在容器中找组件
     */
    @Bean
    public JdbcTemplate jdbcTemplate() throws Exception {
        JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource());
        return jdbcTemplate;
    }
}
/**
 * @author keying
 */
@Repository
public class UserDao {
    @Autowired
    private JdbcTemplate jdbcTemplate;
    public void insert() {
        String sql = "insert into user (uid,name,age) values (?,?,?)";
        String name = UUID.randomUUID().toString().substring(0, 3);
        jdbcTemplate.update(sql, "111", name, 12);
        System.out.println("插入完成");
        //当有异常时,事务让整个方法回滚
        int i = 10/0;
    }
}
/**
 * @author keying
 */
@Service
public class UserService {
    @Autowired
    private UserDao userDao;
    public void inert(){
        userDao.insert();
    }
}

junitTest加载我们的数据库配置类,并且从容器中获取到userService,调用autowired注入的dao,访问操作数据。

/**
 * @author keying
 */
public class TestTx {
    @Test
    public void test() {
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(TxConfig.class);
        UserService userService = applicationContext.getBean(UserService.class);
        userService.inert();
        applicationContext.close();
    }
}

从结果打印可以看到,即使后面的运算抛出异常,但数据库依然插入成功,方法不会回滚。


二、事务开启


事务开启分位三步,缺一不可:


1、@EnableTransactionManagement开启事务。

2、给方法加上@Transactional注解。

3、配置事务管理控制事务。(注意事务管理要交给ioc容器管理)

/**
 * 事务配置类
 * 环境搭建:数据源,数据库、spring-jdbc
 * 配置数据源:JdbcTemplate操作数据(spring提供的操作数据工具)
 *
 * 1、给方法上加上@Transactional
 * 2、给配置文件加上@EnableTransactionManagement开启事务管理
 * 3、配置事务管理器控制事务
 *
 * @author keying
 */
@EnableTransactionManagement
@Configuration
@ComponentScan("com.alibaba.tx")
public class TxConfig {
    @Bean
    public DataSource dataSource() throws Exception {
        ComboPooledDataSource dataSource = new ComboPooledDataSource();
        dataSource.setDriverClass("com.mysql.jdbc.Driver");
        dataSource.setJdbcUrl("jdbc:mysql://127.0.0.1:3306/testmac");
        dataSource.setUser("root");
        dataSource.setPassword("root");
        return dataSource;
    }
    /**
     * spring对@Configuration有特殊处理,给容器中加组件的方法,多次调用也只是在容器中找组件
     */
    @Bean
    public JdbcTemplate jdbcTemplate() throws Exception {
        JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource());
        return jdbcTemplate;
    }
    /**
     * 事务管理器
     */
    @Bean
    public PlatformTransactionManager transactionManager() throws Exception {
        return new DataSourceTransactionManager(dataSource());
    }
}

dao层加个@Transactional注解:

/**
 * @author keying
 */
@Repository
public class UserDao {
    @Autowired
    private JdbcTemplate jdbcTemplate;
    @Transactional
    public void insert() {
        String sql = "insert into user (uid,name,age) values (?,?,?)";
        String name = UUID.randomUUID().toString().substring(0, 3);
        jdbcTemplate.update(sql, "111", name, 12);
        System.out.println("插入完成");
        //当有异常时,事务让整个方法回滚
        int i = 10 / 0;
        try {
        } catch (Exception e) {
        }
    }
}

开启之后可以看到,当运算异常时,数据库并不会插入数据成功。

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
71 2
|
1月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
19天前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
38 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
17天前
|
缓存 安全 Java
Spring高手之路26——全方位掌握事务监听器
本文深入探讨了Spring事务监听器的设计与实现,包括通过TransactionSynchronization接口和@TransactionalEventListener注解实现事务监听器的方法,并通过实例详细展示了如何在事务生命周期的不同阶段执行自定义逻辑,提供了实际应用场景中的最佳实践。
40 2
Spring高手之路26——全方位掌握事务监听器
|
16天前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
38 2
|
19天前
|
Java 关系型数据库 数据库
京东面试:聊聊Spring事务?Spring事务的10种失效场景?加入型传播和嵌套型传播有什么区别?
45岁老架构师尼恩分享了Spring事务的核心知识点,包括事务的两种管理方式(编程式和声明式)、@Transactional注解的五大属性(transactionManager、propagation、isolation、timeout、readOnly、rollbackFor)、事务的七种传播行为、事务隔离级别及其与数据库隔离级别的关系,以及Spring事务的10种失效场景。尼恩还强调了面试中如何给出高质量答案,推荐阅读《尼恩Java面试宝典PDF》以提升面试表现。更多技术资料可在公众号【技术自由圈】获取。
|
16天前
|
Java 数据库连接 数据库
从入门到精通---深入剖析Spring DAO
在Java企业级开发中,Spring框架以其强大的功能和灵活性,成为众多开发者的首选。Spring DAO(Data Access Object)作为Spring框架中处理数据访问的重要模块,对JDBC进行了抽象封装,极大地简化了数据访问异常的处理,并能统一管理JDBC事务。本文将从概述、功能点、背景、业务点、底层原理等多个方面深入剖析Spring DAO,并通过多个Java示例展示其应用实践,同时指出对应实践的优缺点。
21 1
|
20天前
|
监控 Java 数据安全/隐私保护
如何用Spring Boot实现拦截器:从入门到实践
如何用Spring Boot实现拦截器:从入门到实践
38 5
|
1月前
|
Java 开发者 Spring
Spring高手之路24——事务类型及传播行为实战指南
本篇文章深入探讨了Spring中的事务管理,特别是事务传播行为(如REQUIRES_NEW和NESTED)的应用与区别。通过详实的示例和优化的时序图,全面解析如何在实际项目中使用这些高级事务控制技巧,以提升开发者的Spring事务管理能力。
52 1
Spring高手之路24——事务类型及传播行为实战指南
|
23天前
|
JavaScript Java 关系型数据库
Spring事务失效的8种场景
本文总结了使用 @Transactional 注解时事务可能失效的几种情况,包括数据库引擎不支持事务、类未被 Spring 管理、方法非 public、自身调用、未配置事务管理器、设置为不支持事务、异常未抛出及异常类型不匹配等。针对这些情况,文章提供了相应的解决建议,帮助开发者排查和解决事务不生效的问题。
下一篇
DataWorks