详细分析Java中的@Transactional注解

目录

  • 前言
  • [1. 基本知识](#1. 基本知识)
  • [2. 常用属性](#2. 常用属性)
  • [3. Demo](#3. Demo)
  • [4. 总结](#4. 总结)

前言

@Transactional 是 Spring 框架中用于管理事务的注解。

  1. 该注解来源于Spring,对于Spring的基础知识可看我之前的文章:
    Spring框架从入门到学精(全)

  2. 该注解也可用在xxl-job框架中,让事务进行回滚执行,可看我之前的文章:
    详细分析Java中的分布式任务调度框架 XXL-Job

  3. 涉及的事务相关知识可参考之前这篇文章:
    数据库关于事务的详解分析(全)包含面试常问的细节

1. 基本知识

@Transactional 注解用于标记一个方法或类需要被 Spring 托管的事务管理。

它可以应用于类级别的和方法级别的,用于控制事务的行为。

作用 优点 缺点
1.事务管理: 该注解确保被注解的方法或类在执行时将被包装在一个事务中。 2.事务传播: 它定义了在嵌套调用中,新事务是如何与现有事务交互的。 1.简化事务管理: 通过注解方式,简化了对事务的管理,不再需要手动编写事务相关的代码。 2.减少样板代码: 提供了一种声明式的方式,减少了样板式的事务管理代码。 过度使用可能导致性能问题: 在某些情况下,过度使用事务注解可能导致性能下降,因为每个被注解的方法都会被包装在一个事务中。

对于@Transactional注解有好些属性,可通过源码查看:

2. 常用属性

常用的属性主要如下:

一、propagation : 事务的传播行为,默认值是 REQUIRED

常用的取值包括:

  • REQUIRED:如果当前存在事务,则加入该事务;否则,创建一个新事务。
  • REQUIRES_NEW:创建一个新的事务,并挂起当前事务(如果存在)。
  • SUPPORTS:支持当前事务,如果没有事务,则以非事务方式执行。
  • MANDATORY:强制要求存在当前事务,如果不存在,则抛出异常

示例代码如下:

java 复制代码
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void methodWithNewTransaction() {
    // ...
}

二、isolation : 事务的隔离级别,默认是 DEFAULT

常用的取值包括:

  • DEFAULT:使用数据库默认的隔离级别。
  • READ_UNCOMMITTED:允许读取未提交的数据更改。
  • READ_COMMITTED:只能读取已提交的数据更改。
  • REPEATABLE_READ:可重复读,确保在同一事务中对相同数据的多次读取是一致的。
  • SERIALIZABLE:最高隔离级别,确保在同一事务中对相同数据的多次读取和写入都是一致的。

示例代码如下:

java 复制代码
@Transactional(isolation = Isolation.READ_COMMITTED)
public void methodWithCustomIsolationLevel() {
    // ...
}

三、readOnly : 指定事务是否为只读,默认值为 false。

如果设置为 true,表示只读事务,不允许有写操作。

java 复制代码
@Transactional(readOnly = true)
public void readOnlyMethod() {
    // ...
}

四、timeout : 指定事务超时时间,单位为秒。

如果事务执行时间超过设定的超时时间,将回滚事务。

java 复制代码
@Transactional(timeout = 30)
public void methodWithTimeout() {
    // ...
}

五、rollbackFor 和 noRollbackFor: 指定在哪些异常情况下回滚事务或不回滚事务。

java 复制代码
@Transactional(rollbackFor = CustomException.class)
public void methodWithRollbackForException() {
    // ...
}

@Transactional(noRollbackFor = AnotherException.class)
public void methodWithNoRollbackForException() {
    // ...
}

这些是 @Transactional 注解中一些常用的属性。通过设置这些属性,你可以根据具体需求调整事务的行为。

3. Demo

假设有一个简单的银行应用,有两个服务类,一个是转账服务 TransferService,另一个是用户服务 UserService。

确保转账和更新用户余额这两个操作在同一个事务中。

代码如下:

java 复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service
public class TransferService {

    @Autowired
    private UserService userService;

    @Transactional
    public void transferMoney(String fromAccount, String toAccount, double amount) {
        // 扣除转账账户余额
        userService.decreaseBalance(fromAccount, amount);

        // 增加接收账户余额
        userService.increaseBalance(toAccount, amount);
    }
}

以及

java 复制代码
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

@Service
public class UserService {

    @Autowired
    private JdbcTemplate jdbcTemplate;

    @Transactional
    public void decreaseBalance(String account, double amount) {
        // 扣除余额的数据库更新操作
        // ...
    }

    @Transactional
    public void increaseBalance(String account, double amount) {
        // 增加余额的数据库更新操作
        // ...
    }
}

在上述示例中,@Transactional 注解确保了 transferMoney 方法和 decreaseBalance、increaseBalance 方法都在同一个事务中执行。

如果其中任何一个方法发生异常,整个事务将回滚。

对于实际的应用场景,一个rollbackFor也可:@Transactional(rollbackFor = Exception.class)

具体如下:

也可配合XXL-Job的框架进行使用,主要如下:

4. 总结

对应的场景案例可看这篇文章(个人感觉不错):spring中@Transactional注解的作用,使用场景举例

  1. @Transactional 注解只能用在public 方法上,private以及protected不会报错,但不会生效
  2. 该注解只有在spring容器中扫描到才生效
  3. 可以用在类上或者方法上,范围不一致而已
  4. 如果使用了try catch,注解会失效。(虽然只能回滚非检查型异常,具体为RuntimeException及其子类和Error子类,但要想捕获可以尝试加入@Transactional(rollbackFor = Exception.class)
  5. 非事务方法调用事务方法,要用代理对象调用否则事务会失效。而事务方法调用非事务不会失效。
相关推荐
qmx_0720 分钟前
HTB-Jerry(tomcat war文件、msfvenom)
java·web安全·网络安全·tomcat
为风而战28 分钟前
IIS+Ngnix+Tomcat 部署网站 用IIS实现反向代理
java·tomcat
技术无疆2 小时前
快速开发与维护:探索 AndroidAnnotations
android·java·android studio·android-studio·androidx·代码注入
架构文摘JGWZ5 小时前
Java 23 的12 个新特性!!
java·开发语言·学习
拾光师6 小时前
spring获取当前request
java·后端·spring
aPurpleBerry6 小时前
neo4j安装启动教程+对应的jdk配置
java·neo4j
我是苏苏6 小时前
Web开发:ABP框架2——入门级别的增删改查Demo
java·开发语言
xujinwei_gingko6 小时前
Spring IOC容器Bean对象管理-Java Config方式
java·spring
2301_789985946 小时前
Java语言程序设计基础篇_编程练习题*18.29(某个目录下的文件数目)
java·开发语言·学习
IT学长编程7 小时前
计算机毕业设计 教师科研信息管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解
java·毕业设计·springboot·毕业论文·计算机毕业设计选题·计算机毕业设计开题报告·教师科研管理系统