面试官:Spring@Transactional注解在什么情况下事务不生效?

本文涉及的产品
可观测可视化 Grafana 版,10个用户账号 1个月
可观测监控 Prometheus 版,每月50GB免费额度
应用实时监控服务-用户体验监控,每月100OCU免费额度
简介: 这篇笔记来学习一下使用Spring框架的时候,@Transactional注解标注的方法在什么情况下事务不会生效。我们可以写一个demo项目,引入以下依赖

这篇笔记来学习一下使用Spring框架的时候,@Transactional注解标注的方法在什么情况下事务不会生效。

我们可以写一个demo项目,

引入以下依赖

<dependencies>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.6</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>5.2.8.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.2.8.RELEASE</version>
        </dependency>
    </dependencies>

项目的目录结构如下:

我们新建一个user表,之后会用上

CREATE TABLE `tb_user` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `name` varchar(255) NOT NULL DEFAULT '',
  `age` int(11) NOT NULL DEFAULT '0',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=50 DEFAULT CHARSET=utf8

然后我们需要在 site.nemo.entity 包里面定义一个User类:

package site.nemo.entity;
public class User {
    private Integer id;
    private String name;
    private Integer age;
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Integer getAge() {
        return age;
    }
    public void setAge(Integer age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

然后我们还需要一个Dao,来操作user表。在 site.nemo.dao 包下新建一个UserDao类:

@Repository
public class UserDao {
    @Autowired
    private JdbcTemplate jdbcTemplate;
    public void save(User user) {
        jdbcTemplate.update("insert into tb_user (name, age) values (?, ?)", user.getName(), user.getAge());
    }
}

然后我们还需要对jdbc的数据源进行一些配置,在 site.nemo.configuration 包里面新建一个配置类:

@Configuration
@ComponentScan(basePackages = {"site.nemo.service", "site.nemo.dao"})
public class TransactionConfiguration {
    @Bean
    public JdbcTemplate jdbcTemplate() {
        return new JdbcTemplate(dataSource());
    }
    @Bean
    public DataSource dataSource() {
        DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setDriverClassName("com.mysql.jdbc.Driver");
        dataSource.setUrl("jdbc:mysql://xxx.xxx.xxx.xxx:3306/mytestdb?characterEncoding=utf8");
        dataSource.setUsername("xxxx");
        dataSource.setPassword("xxxx");
        return dataSource;
    }
}

最后我们在 site.nemo.service 里面新建一个UserService:

@Service
public class UserService {
    @Autowired
    private UserDao userDao;
}

我们下面开始分析几个@Transactional没有起作用的原因

原因一:没有开启事务管理

我们给 UserService 添加一个方法,如下所示。

@Transactional
    public void save1() {
        User user = new User();
        user.setName("1");
        user.setAge(1);
        userDao.save(user);
        int i = 1 / 0;
    }

该方法是往tb_user表里面插入一条数据,在方法的最后有一个 1/0 ,肯定会报错。我希望的是,在该方法上标注了 @Transactional 注解,当它里面有异常的时候,能够事务回滚。也就是希望数据没有被插入数据库。

我们现在在 site.nemo 包下新建一个类 Application1 ,来调用 UserServicesave1 方法

public class Application1 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save1();
    }
}

运行 Application1 的main方法,可以看到控制台打印出了错误堆栈。

但是, 1 这个用户还是被插入到数据库了。并没有事务回滚。

原因是,我们并没有添加 @
EnableTransactionManagement
来开启事务管理,所以 @Transactional 没生效。

当我们在 TransactionConfiguration 这个类上面加上 @
EnableTransactionManagement
注解之后,再执行 Application1 的main方法,可以看到数据没有插入即事务被回滚了。

我们知道如果使用的是springboot框架,它自动帮你开启了事务管理。可以从框架源码里面看一下

springboot框架的启动类都会加上 @SpringBootApplication 注解,而 @SpringBootApplication注解其实是 @EnableAutoConfiguration 注解和其他注解的组合。

@EnableAutoConfiguration 注解使用了@Import来引入
AutoConfigurationImportSelector.class
选择器。(关于如何使用@Import和ImportSelector来实现模块注解,可以看我的另外一篇笔记)

查看
AutoConfigurationImportSelector.class
的源码,在 selectImports 方法中找到了 getAutoConfigurationEntry 方法的调用。

进入 getAutoConfigurationEntry 方法看一下,它调用了
getCandidateConfigurations
方法

进入
getCandidateConfigurations
方法,可以看到它会去读spring.factories文件

可以知道
getCandidateConfigurations
方法会去读 spring.factories 文件,我们可以从项目的 Externnal Libraries 里面找到
org.springframework.boot:spring-boot-autoconfigure
,找到它的 META-INF 文件夹,可以看到里面有 spring.factories 文件

点开 spring.factories 文件,搜索一下与transaction有关的。

我们点进
TransactionAutoConfiguration
类里面去看一下,从它的注解看出来,它就是自动装配事务的

看一下
TransactionAutoConfiguration
类里面的具体内容,看到了我们熟悉的 @
EnableTransactionManagement
注解。

除了上面说的在普通的非springboot项目里面没有开启了事务管理 @
EnableTransactionManagement
这个原因导致@Transactional没有生效,下面也会分析以下其他原因。下面分析的几个case,都是在已经开启了事务管理 @
EnableTransactionManagement
的基础上。

原因二:标注了@Transactional的方法里面的异常被捕获了

我们给 UserService 再添加一个 save2 方法,如下所示。

@Transactional
    public void save2() {
      try {
            User user = new User();
          user.setName("2");
          user.setAge(2);
          userDao.save(user);
          int i = 1 / 0;
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

我们现在在 site.nemo 包下新建一个类 Application2 ,来调用 UserServicesave2 方法

public class Application2 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save2();
    }
}

从结果可知,事务没有回滚。

因为save2方法使用了try/catch捕获了异常,所以即使标注了@Transactional,这个方法也还是没回滚。

我们可以点进去@Transactional看下它的具体内容,可以看到它有一个 rollbackFor 属性。看下它的注释

注释里面提到,默认情况下,当发生 Error 或者 RuntimeException 的时候,才会回滚。

而我们的save2方法上面标注的@Transactional并没有指定 rollbackFor 属性,而且save2里面的异常被我们捕获了且没有再抛出来,所以save2没有回滚。

原因三:标注了@Transactional的方法发生了非 Error 或者 RuntimeException

从第二个原因里面可以知道,默认情况下,当发生 Error 或者 RuntimeException 的时候,才会回滚。

所以我们来试一下当抛出的不是这两个异常的时候,会不会回滚

我们可以自定义一个 BusinessException ,它继承的是Exception

package site.nemo.exceptions;
public class BusinessException extends Exception {
    public BusinessException() {
        super();
    }
    public BusinessException(String msg) {
        super(msg);
    }
}

然后在 UserService 里面再添加一个方法 save3

@Transactional
    public void save3() throws BusinessException {
        User user = new User();
        user.setName("3");
        user.setAge(3);
        userDao.save(user);
        throw new BusinessException("test save3");
    }

我们现在在 site.nemo 包下新建一个类 Application3 ,来调用 UserServicesave3 方法

public class Application3 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        try {
            userService.save3();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

从结果可知,事务没有回滚。

关于java里面的unchecked和checked异常的区别,可以看我的另外一篇笔记

原因四:标注了@Transactional的方法的事务传播类型propagation配置成了NOTSUPPORT

事务的传播类型见我的另一篇笔记

NOTSUPPORT表示不支持事务,即使当前有事务,也不会使用事务。

所以当 propagation = Propagation.NOT_SUPPORTED 的时候,不会使用事务。所以异常发生的时候,也就不会回滚。

我们可以试验一下, 在 UserService 里面添加 save4 方法,在它上面声明@Transactional注解,并且设置 propagation = Propagation.NOT_SUPPORTED

@Transactional(propagation = Propagation.NOT_SUPPORTED)
    public void save4() {
        User user = new User();
        user.setName("4");
        user.setAge(4);
        userDao.save(user);
        int i = 1 / 0;
    }

我们现在在 site.nemo 包下新建一个类 Application4 ,来调用 UserServicesave4 方法

public class Application4 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save4();
    }
}

从结果可知,没有回滚。

原因五:标注了@Transactional的方法的事务传播类型propagation配置成了NEVER

NEVER表示不支持事务,如果有事务则会报错

我们可以试验一下, 在 UserService 里面添加 save5 方法,在它上面声明@Transactional注解,并且设置 propagation = Propagation.NEVER

@Transactional(propagation = Propagation.NEVER)
    public void save5() {
        User user = new User();
        user.setName("5");
        user.setAge(5);
        userDao.save(user);
        int i = 1 / 0;
    }

我们现在在 site.nemo 包下新建一个类 Application5 ,来调用 UserServicesave5 方法

public class Application5 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save5();
    }
}

从结果可知,没有回滚。

原因六:标注了@Transactional的方法的事务传播类型propagation配置成了SUPPORTS且当前没有事务

SUPPORTS的意思是,如果当前有事务,就加入,如果没事务,则以非事务运行。

我们可以试验一下, 在 UserService 里面添加 save6 方法,在它上面声明@Transactional注解,并且设置 propagation = Propagation.SUPPORTS

@Transactional(propagation = Propagation.SUPPORTS)
    public void save6() {
        User user = new User();
        user.setName("6");
        user.setAge(6);
        userDao.save(user);
        int i = 1 / 0;
    }

我们现在在 site.nemo 包下新建一个类 Application6 ,来调用 UserServicesave6 方法

public class Application6 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save6();
    }
}

从结果可知,没有回滚。

原因七:外部调用方法A,A内部调用方法B,A没有@Transaction注解而B有@Transactional注解

UserService 里面添加 save7 方法,和 save72 方法,其中 save72 上面标有@Transactional注解,且save72里面有异常

public void save7() {
        User user = new User();
        user.setName("7");
        user.setAge(7);
        userDao.save(user);
        save72();
    }
    @Transactional
    public void save72() {
        User user = new User();
        user.setName("72");
        user.setAge(72);
        userDao.save(user);
        int i = 1 / 0;
    }

我们现在在 site.nemo 包下新建一个类 Application7 ,来调用 UserServicesave7 方法

public class Application7 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save7();
    }
}

从结果可以看出,save72方法没有回滚。

这是因为, save7 方法没有标注@Transactional注解,它内部调用 save72() 其实可以看做是 this.save72() ,这里的、this其实是个普通对象,没有被AOP动态代理增强过。所以 save72()出现异常的时候没有回滚。

那么其实我们也可以知道,如果save7和sav72上面都有@Transactional注解的话,事务最终会回滚,并不是因为save72上面的注解生效了,而是因为save7上面的注解生效了,save72回滚只不过是因为被包在了save7的事务里面,是在整个大事务里面回滚的。

原因八:标注了@Transactional的方法A的propagation配置成了REQUIRE,标注了@Transactional的方法B的propagation配置成了REQUIRE_NEW,方法A调用了方法B

REQUIRE表示如果当前有事务,则加入事务;如果当前没事务,则新起一个事务;

REQUEIRE_NEW表示不管当前是否有事务,都新起一个事务

标注了@Transactional的方法A的propagation配置成了REQUIRE,标注了@Transactional的方法B的propagation配置成了REQUIRE_NEW,方法A调用了方法B。那么当A最后因为异常回滚的时候,B不会回滚。

实验一下,可以新建一个 UserService2 ,里面有一个 save8() ,它的事务传播类型是 REQUIREDUserService2 有一个属性是 UserService 的实例。在 UserService2#save8() 里面会调用 UserService#save82() 方法。

@Service
public class UserService2 {
    @Autowired
    private UserDao userDao;
    @Autowired
    private UserService userService;
    @Transactional(propagation = Propagation.REQUIRED)
    public void save8() {
        User user = new User();
        user.setName("service2-8");
        user.setAge(8);
        userDao.save(user);
        userService.save82();
        int i = 1/0;
    }
}

UserService 里面添加方法 save82() ,它的事务传播类型是 REQUIRES_NEW

@Transactional(propagation = Propagation.REQUIRES_NEW)
    public void save82() {
        User user = new User();
        user.setName("82");
        user.setAge(82);
        userDao.save(user);
        int i = 1 / 0;
    }

我们现在在 site.nemo 包下新建一个类 Application8 ,来调用 UserServicesave8 方法

public class Application8 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService2 userService2 = context.getBean(UserService2.class);
        userService2.save8();
    }
}

从结果可以看出,save8回滚了,但是save82没有回滚。因为save82的事务传播类型是REQUIRES_NEW,它会新起一个事务,与原来的事务没关系。

原因九:标注了@Transactional的方法不是public的

可以在 UserService 里面新增一个protect方法 save9

@Transactional
    protected void save9() {
        User user = new User();
        user.setName("9");
        user.setAge(9);
        userDao.save(user);
        int i = 1 / 0;
    }

由于protect方法只能包内调用,我们可以在 site.nemo.service 里面加入一个 UserServiceManager 类,该类调用了 UserService 的protect方法

package site.nemo.service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
@Component
public class UserManager {
    @Autowired
    private UserService userService;
    public void testUserService9() {
        userService.save9();
    }
}

我们现在在 site.nemo 包下新建一个类 Application9

public class Application9 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserManager userManager = context.getBean(UserManager.class);
        userManager.testUserService9();
    }
}

从结果可以看出,没有回滚。

原因十:标注了@Transactional的方法发生的异常不是rollbackFor指定的类型或子类

可以在 UserService 里面新增一个方法 save10 ,它的@Transactional里指定rollbackFor为BusinessException,即发生BusinessException或它子类异常的时候才会回滚。但是我在 save10 里面抛出的是Exception类型的异常。

@Transactional(rollbackFor = BusinessException.class)
    public void save10() throws Exception {
        User user = new User();
        user.setName("10");
        user.setAge(10);
        userDao.save(user);
        throw new Exception("test save10");
    }

我们现在在 site.nemo 包下新建一个类 Application10

public class Application10 {
    public static void main(String[] args) {
        ApplicationContext context = new AnnotationConfigApplicationContext(TransactionConfiguration.class);
        UserService userService = context.getBean(UserService.class);
        userService.save10();
    }
}

从结果可以看出,没有回滚。

原因十一:数据库不支持事务

如果数据库引擎不支持事务的话,随便怎么加@Transactional,都不会生效。

本文就是愿天堂没有BUG给大家分享的内容,大家有收获的话可以分享下,想学习更多的话可以到微信公众号里找我,我等你哦。

相关实践学习
如何快速连接云数据库RDS MySQL
本场景介绍如何通过阿里云数据管理服务DMS快速连接云数据库RDS MySQL,然后进行数据表的CRUD操作。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
15天前
|
XML Java 测试技术
Spring IOC—基于注解配置和管理Bean 万字详解(通俗易懂)
Spring 第三节 IOC——基于注解配置和管理Bean 万字详解!
98 26
|
18天前
|
缓存 Java 数据库
SpringBoot缓存注解使用
Spring Boot 提供了一套方便的缓存注解,用于简化缓存管理。通过 `@Cacheable`、`@CachePut`、`@CacheEvict` 和 `@Caching` 等注解,开发者可以轻松地实现方法级别的缓存操作,从而提升应用的性能和响应速度。合理使用这些注解可以大大减少数据库的访问频率,优化系统性能。
163 89
|
2月前
|
Java Spring
【Spring】方法注解@Bean,配置类扫描路径
@Bean方法注解,如何在同一个类下面定义多个Bean对象,配置扫描路径
181 73
|
5天前
|
监控 Java Spring
SpringBoot:SpringBoot通过注解监测Controller接口
本文详细介绍了如何通过Spring Boot注解监测Controller接口,包括自定义注解、AOP切面的创建和使用以及具体的示例代码。通过这种方式,可以方便地在Controller方法执行前后添加日志记录、性能监控和异常处理逻辑,而无需修改方法本身的代码。这种方法不仅提高了代码的可维护性,还增强了系统的监控能力。希望本文能帮助您更好地理解和应用Spring Boot中的注解监测技术。
33 16
|
2月前
|
Java Spring 容器
【SpringFramework】Spring IoC-基于注解的实现
本文主要记录基于Spring注解实现IoC容器和DI相关知识。
60 21
|
1月前
|
SQL Java 关系型数据库
【SpringFramework】Spring事务
本文简述Spring中数据库及事务相关衍伸知识点。
50 9
|
2月前
|
Java 开发者 Spring
理解和解决Spring框架中的事务自调用问题
事务自调用问题是由于 Spring AOP 代理机制引起的,当方法在同一个类内部自调用时,事务注解将失效。通过使用代理对象调用、将事务逻辑分离到不同类中或使用 AspectJ 模式,可以有效解决这一问题。理解和解决这一问题,对于保证 Spring 应用中的事务管理正确性至关重要。掌握这些技巧,可以提高开发效率和代码的健壮性。
128 13
|
2月前
|
存储 Java Spring
【Spring】获取Bean对象需要哪些注解
@Conntroller,@Service,@Repository,@Component,@Configuration,关于Bean对象的五个常用注解
|
2月前
|
Java Spring
【Spring配置】idea编码格式导致注解汉字无法保存
问题一:对于同一个项目,我们在使用idea的过程中,使用汉字注解完后,再打开该项目,汉字变成乱码问题二:本来a项目中,汉字注解调试好了,没有乱码了,但是创建出来的新的项目,写的注解又成乱码了。
|
2月前
|
缓存 安全 Java
Spring高手之路26——全方位掌握事务监听器
本文深入探讨了Spring事务监听器的设计与实现,包括通过TransactionSynchronization接口和@TransactionalEventListener注解实现事务监听器的方法,并通过实例详细展示了如何在事务生命周期的不同阶段执行自定义逻辑,提供了实际应用场景中的最佳实践。
76 2
Spring高手之路26——全方位掌握事务监听器