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

本文涉及的产品
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) {
        }
    }
}

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

相关实践学习
基于CentOS快速搭建LAMP环境
本教程介绍如何搭建LAMP环境,其中LAMP分别代表Linux、Apache、MySQL和PHP。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
6天前
|
小程序 数据可视化 Java
Java+后端Spring boot 开发的全套UWB定位方案,0.1米高精度定位系统源码
UWB定位系统由硬件定位设备、定位引擎和应用软件组成。该定位系统应用软件支持PC端和移动端访问,并提供位置实时显示、历史轨迹回放、人员考勤、电子围栏、行为分析、智能巡检等功能。定位精度高达10cm,同时具备高动态、高容量、低功耗的优点。应用场景包括:隧道、化工、工厂、煤矿、工地、电厂、养老、展馆、整车、机房、机场等。
31 8
|
2天前
|
监控 Java API
【监控】spring actuator源码速读
【监控】spring actuator源码速读
6 1
|
2天前
|
监控 Java 关系型数据库
java版MES系统源码,后端采用 Spring Boot 多模块架构
MES系统采用Vue3的vue-element-plus-admin为后台,Spring Boot多模块架构,支持MySQL、Oracle等数据库,具备SaaS多租户功能。核心功能包括车间计划排程、工艺流程配置、生产质量管理、进度追踪、库存和排班管理等,全面覆盖生产运营关键环节。
java版MES系统源码,后端采用 Spring Boot 多模块架构
|
2天前
|
Java Spring
Spring源码学习——(二)
第二讲——了解BeanFactory的功能
|
3天前
|
Java Spring 容器
Spring源码学习——(一)
第一讲——了解BeanFactory和ApplicationContext
|
3天前
|
Java 开发者 Spring
spring事务类型,事务传递,隔离级别?
spring事务类型,事务传递,隔离级别?
|
5天前
|
Java Spring 容器
Spring 中一个事务方法调用另一个事务方法会怎样?
【6月更文挑战第5天】如果我们在一个包含事务的方法中,调用了另一个包含里事务的方法,那么,Spring 会创建新的事物吗?如果方法抛出异常导致事务会滚了呢?
17 0
|
7天前
|
XML 存储 Java
Spring 6(一)【Spring 入门】
Spring 6(一)【Spring 入门】
|
9天前
|
Java 数据库连接 数据库
Spring日志完结篇,MyBatis操作数据库(入门)
Spring日志完结篇,MyBatis操作数据库(入门)
|
10天前
|
存储 自动驾驶 Java
Spring IoC&DI(1)—入门
Spring IoC&DI(1)—入门
11 1