Java事务失效(面试题)的常见场景

1. 方法非public修饰

原理: Spring AOP代理(CGLIB或JDK动态代理)默认无法拦截非public方法。

示例:

java 复制代码
@Service
public class UserService {
    @Transactional
    void updateUser() { // 非public方法
        // 事务不会生效!
    }
}

修复: 将方法改为public

2. 自调用(同一个类内部调用)

原理: 事务基于AOP代理,自调用绕过代理直接调用目标方法。
示例:

java 复制代码
@Service
public class OrderService {
    public void placeOrder() {
        deductStock(); // 直接内部调用,事务失效
    }

    @Transactional
    public void deductStock() {
        // 事务不生效!
    }
}

修复:

  • 从其他Bean注入自身(通过@Autowired注入代理对象)

  • 使用AopContext.currentProxy()(需开启exposeProxy

3. 异常类型错误或被捕获

场景1:抛出非RuntimeException/Error
原理: 默认只回滚RuntimeExceptionError
示例:

java 复制代码
@Transactional
public void update() throws IOException {
    if (error) throw new IOException(); // 检查型异常,不回滚!
}

修复:

添加rollbackFor属性

java 复制代码
@Transactional(rollbackFor = Exception.class)

场景2:异常被catch后未重新抛出
原理: 事务管理器检测不到异常。
示例:

java 复制代码
@Transactional
public void update() {
    try {
        db.update(...); // 抛出SQLException
    } catch (Exception e) {
        // 捕获后不抛出,事务提交!
    }
}

修复:catch中抛出RuntimeException

java 复制代码
catch (Exception e) {
    throw new RuntimeException(e);
}

4. 多线程环境下事务上下文丢失

原理: Spring事务信息存储在ThreadLocal中,新线程无法继承上下文。
示例:

java 复制代码
@Transactional
public void asyncUpdate() {
    new Thread(() -> {
        userDao.update(); // 子线程操作无事务控制!
    }).start();
}

修复:

使用Spring的@Async + Transactional(需配置异步任务执行器)

java 复制代码
@Async
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void asyncTask() {
    // 新事务生效
}

5. 数据库引擎不支持事务

原理: 如MySQL的MyISAM引擎不支持事务。
检查:

java 复制代码
SHOW TABLE STATUS LIKE 'table_name'; -- 查看Engine类型

修复: 改用InnoDB引擎。

6. 嵌套事务传播配置错误

场景: 内层事务使用Propagation.NOT_SUPPORTED
示例:

java 复制代码
@Transactional
public void outer() {
    inner();
}

@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void inner() {
    // 挂起外层事务,操作无事务保护!
}

7. 非Spring管理Bean中使用@Transactional

原理: @Transactional由Spring代理实现。
错误示例:

java 复制代码
public class Util {
    @Transactional // 未被Spring管理,注解无效
    public static void save() { ... }
}
相关推荐
廿一夏6 小时前
MySql存储引擎与索引
数据库·sql·mysql
Mahir086 小时前
Spring 循环依赖深度解密:从问题本质到三级缓存源码级解析
java·后端·spring·缓存·面试·循环依赖·三级缓存
RyFit7 小时前
SpringAI 常见问题及解决方案大全
java·ai
石山代码7 小时前
C++ 内存分区 堆区
java·开发语言·c++
绝知此事8 小时前
【算法突围 01】线性结构与哈希表:后端开发的收纳术
java·数据结构·算法·面试·jdk·散列表
无风听海8 小时前
C# 隐式转换深度解析
java·开发语言·c#
lzhdim8 小时前
SQL 入门 15:SQL 事务:从 ACID 到四种常见的并发问题
数据库·sql
瀚高PG实验室8 小时前
瀚高企业版V9.1.1在pg_restore还原备份文件时提示extract函数语法问题
数据库·瀚高数据库
一只大袋鼠8 小时前
Git 进阶(二):分支管理、暂存栈、远程仓库与多人协作
java·开发语言·git
TDengine (老段)8 小时前
TDengine Tag 设计哲学与 Schema 变更机制
大数据·数据库·物联网·时序数据库·iot·tdengine·涛思数据