当前位置: 代码网 > it编程>编程语言>Java > Spring 事务管理三种实现方式详解

Spring 事务管理三种实现方式详解

2026年05月08日 Java 我要评论
一、前言在上一篇博客中,我们学习了 spring jdbctemplate 实现转账操作。但是:如果转账过程中程序报错,钱会只扣不加,造成数据错误!所以必须使用 spring 事务管理,保证转账要么全

一、前言

在上一篇博客中,我们学习了 spring jdbctemplate 实现转账操作。但是:如果转账过程中程序报错,钱会只扣不加,造成数据错误!

所以必须使用 spring 事务管理,保证转账要么全部成功,要么全部失败。本文讲解 spring 事务 3 种配置方式,基于转账案例,通俗易懂,适合新手学习。

二、事务核心 api

spring 事务管理的核心接口:

  1. platformtransactionmanager:平台事务管理器
  2. datasourcetransactionmanager:jdbc / mybatis 专用实现类
  3. transactiondefinition:事务定义(隔离级别、传播行为)

三、环境说明

 maven 依赖(pom.xml)

<?xml version="1.0" encoding="utf-8"?>
<project xmlns="http://maven.apache.org/pom/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/xmlschema-instance"
         xsi:schemalocation="http://maven.apache.org/pom/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelversion>4.0.0</modelversion>
    <groupid>com.qcby</groupid>
    <artifactid>spring-aop-demo</artifactid>
    <version>1.0-snapshot</version>
    <dependencies>
        <!-- spring核心 -->
        <dependency>
            <groupid>org.springframework</groupid>
            <artifactid>spring-context</artifactid>
            <version>5.3.31</version>
        </dependency>
        <dependency>
            <groupid>org.springframework</groupid>
            <artifactid>spring-jdbc</artifactid>
            <version>5.3.31</version>
        </dependency>
        <!-- 注解支持 -->
        <dependency>
            <groupid>javax.annotation</groupid>
            <artifactid>javax.annotation-api</artifactid>
            <version>1.3.2</version>
        </dependency>
        <!-- 日志 -->
        <dependency>
            <groupid>commons-logging</groupid>
            <artifactid>commons-logging</artifactid>
            <version>1.2</version>
        </dependency>
        <dependency>
            <groupid>log4j</groupid>
            <artifactid>log4j</artifactid>
            <version>1.2.12</version>
        </dependency>
        <!-- 测试 -->
        <dependency>
            <groupid>junit</groupid>
            <artifactid>junit</artifactid>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupid>org.springframework</groupid>
            <artifactid>spring-test</artifactid>
            <version>5.3.31</version>
            <scope>test</scope>
        </dependency>
        <!-- 数据库 -->
        <dependency>
            <groupid>com.alibaba</groupid>
            <artifactid>druid</artifactid>
            <version>1.1.10</version>
        </dependency>
        <dependency>
            <groupid>mysql</groupid>
            <artifactid>mysql-connector-java</artifactid>
            <version>8.0.28</version>
        </dependency>
        <!-- aop 相关 -->
        <dependency>
            <groupid>aopalliance</groupid>
            <artifactid>aopalliance</artifactid>
            <version>1.0</version>
        </dependency>
        <dependency>
            <groupid>org.springframework</groupid>
            <artifactid>spring-aspects</artifactid>
            <version>5.3.31</version>
        </dependency>
        <dependency>
            <groupid>org.aspectj</groupid>
            <artifactid>aspectjweaver</artifactid>
            <version>1.8.3</version>
        </dependency>
    </dependencies>
</project>

配置文件:application.properties 存放数据库连接信息

jdbc.driverclassname=com.mysql.cj.jdbc.driver
jdbc.url=jdbc:mysql:///spring_db?usessl=false&servertimezone=utc
jdbc.username=root
jdbc.password=123456

数据库表结构

create table account(
    name varchar(20),
    money double
);
-- 初始化数据
insert into account values ('熊大',1000),('熊二',500);

四、✨ 方式一:xml 声明式事务

1、原理

基于 aop 切面,在 xml 中配置事务规则,对业务方法进行增强。

2、完整 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: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">
    <!-- 加载配置 -->
    <context:property-placeholder location="classpath:application.properties"/>
    <!-- 连接池 -->
    <bean id="datasource" class="com.alibaba.druid.pool.druiddatasource">
        <property name="driverclassname" value="${jdbc.driverclassname}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
    </bean>
    <!-- 1. 事务管理器 -->
    <bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager">
        <property name="datasource" ref="datasource"/>
    </bean>
    <!-- 2. 事务通知 -->
    <tx:advice id="txadvice" transaction-manager="transactionmanager">
        <tx:attributes>
            <tx:method name="pay" isolation="default" propagation="required"/>
            <tx:method name="find*" read-only="true"/>
        </tx:attributes>
    </tx:advice>
    <!-- 3. aop 切面配置 -->
    <aop:config>
        <aop:advisor advice-ref="txadvice" pointcut="execution(* com.qcby.demo4.accountserviceimpl.pay(..))"/>
    </aop:config>
    <!-- dao & service -->
    <bean id="accountdao" class="com.qcby.demo4.accountdaoimpl">
        <property name="datasource" ref="datasource"/>
    </bean>
    <bean id="accountservice" class="com.qcby.demo4.accountserviceimpl">
        <property name="accountdao" ref="accountdao"/>
    </bean>
</beans>

3、业务代码

// service
public class accountserviceimpl implements accountservice {
    private accountdao accountdao;
    public void setaccountdao(accountdao accountdao) {
        this.accountdao = accountdao;
    }
    @override
    public void pay(string out, string in, double money) {
        accountdao.outmoney(out, money);
        // int i = 1 / 0;  开启异常,事务回滚
        accountdao.inmoney(in, money);
    }
}

五、✨ 方式二:xml + 注解事务

1、步骤

  1. 开启注解扫描
  2. 开启事务注解驱动
  3. 在 service 上加 @transactional

2、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.qcby.demo5"/>
    <context:property-placeholder location="classpath:application.properties"/>
    <!-- 连接池 -->
    <bean id="datasource" class="com.alibaba.druid.pool.druiddatasource">
        <property name="driverclassname" value="${jdbc.driverclassname}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
    </bean>
    <!-- 事务管理器 -->
    <bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager">
        <property name="datasource" ref="datasource"/>
    </bean>
    <!-- jdbctemplate -->
    <bean id="jdbctemplate" class="org.springframework.jdbc.core.jdbctemplate">
        <property name="datasource" ref="datasource"/>
    </bean>
    <!-- 开启事务注解 -->
    <tx:annotation-driven transaction-manager="transactionmanager"/>
</beans>

3、service 事务注解

@service
@transactional(isolation = isolation.default)
public class accountserviceimpl implements accountservice {
    @autowired
    private accountdao accountdao;
    @override
    public void pay(string out, string in, double money) {
        accountdao.outmoney(out, money);
        // int a = 1/0;  报错自动回滚
        accountdao.inmoney(in, money);
    }
}

六、✨ 方式三:纯注解事务(完全无 xml)

1、配置类

@configuration
@componentscan("com.qcby.demo6")
@enabletransactionmanagement // 开启事务
public class springconfig {
    @bean
    public datasource datasource(){
        druiddatasource ds = new druiddatasource();
        ds.setdriverclassname("com.mysql.cj.jdbc.driver");
        ds.seturl("jdbc:mysql:///spring_db?usessl=false&servertimezone=utc");
        ds.setusername("root");
        ds.setpassword("zhen@777");
        return ds;
    }
    @bean
    public jdbctemplate jdbctemplate(datasource datasource){
        return new jdbctemplate(datasource);
    }
    @bean
    public platformtransactionmanager transactionmanager(datasource datasource){
        return new datasourcetransactionmanager(datasource);
    }
}

2、service 代码

@service
@transactional
public class accountserviceimpl implements accountservice {
    @autowired
    private accountdao accountdao;
    @override
    public void pay(string out, string in, double money) {
        accountdao.outmoney(out, money);
        accountdao.inmoney(in, money);
    }
}

七、测试代码(通用)

@runwith(springjunit4classrunner.class)
@contextconfiguration("classpath:applicationcontext_tx.xml")
public class demo4test {
    @autowired
    private accountservice accountservice;
    @test
    public void testpay(){
        accountservice.pay("熊大","熊二",100);
        system.out.println("✅ 转账成功(事务已生效)");
    }
}

八、事务三种方式

方式优点
xml 声明式事务无侵入、不修改代码
xml + 注解事务简单易用、最常用
纯注解事务无配置文件、简洁

到此这篇关于spring 事务管理三种实现方式详解的文章就介绍到这了,更多相关spring 事务管理内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2026  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com