Spring?中的切面與事務(wù)結(jié)合使用完整示例
一、前置知識(shí):Spring AOP 與 事務(wù)的關(guān)系
事務(wù)本質(zhì)上就是一個(gè)“切面”
@Transactional注解底層就是通過 AOP + 動(dòng)態(tài)代理 實(shí)現(xiàn)的。- 當(dāng)你在一個(gè)方法上加了
@Transactional,Spring 會(huì):- 創(chuàng)建代理對(duì)象
- 在方法執(zhí)行前開啟事務(wù)
- 方法成功返回時(shí)提交事務(wù)
- 方法拋出異常時(shí)回滾事務(wù)
二、核心組件
| 組件 | 作用 |
|---|---|
PlatformTransactionManager | 事務(wù)管理器,負(fù)責(zé)開啟、提交、回滾事務(wù) |
TransactionDefinition | 事務(wù)定義(隔離級(jí)別、傳播行為等) |
TransactionStatus | 當(dāng)前事務(wù)狀態(tài) |
@Aspect | 定義切面 |
@Around | 環(huán)繞通知,控制方法執(zhí)行全過程 |
三、完整示例:用切面實(shí)現(xiàn)事務(wù)控制
場(chǎng)景:
我們有一個(gè) UserService,其中 createUserWithOrder 方法需要:
- 保存用戶
- 保存訂單
- 如果任一步失敗,整個(gè)操作回滾
我們將不用 @Transactional,而是用一個(gè)切面來(lái)控制事務(wù)。
1. 添加依賴(pom.xml)
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency>
<groupId>com.h2database</groupId>
<artifactId>h2</artifactId>
<scope>runtime</scope>
</dependency>
</dependencies>2. 創(chuàng)建實(shí)體類
// User.java
public class User {
private Long id;
private String name;
}// Order.java
public class Order {
private Long id;
private String productName;
private Long userId;
}3. 創(chuàng)建 Service 方法(標(biāo)記自定義注解)
@Service
public class UserService {
@Autowired
private JdbcTemplate jdbcTemplate;
// 自定義注解,表示此方法需要事務(wù)
@TransactionalAspect
public void createUserWithOrder(String userName, String productName) {
// 1. 保存用戶
String userSql = "INSERT INTO users(name) VALUES(?)";
jdbcTemplate.update(userSql, userName);
// 模擬異常:如果 productName 是 "fail",則拋出異常
if ("fail".equals(productName)) {
throw new RuntimeException("訂單創(chuàng)建失??!");
}
// 2. 保存訂單
String orderSql = "INSERT INTO orders(product_name, user_id) VALUES(?, ?)";
// 假設(shè)用戶ID是自增,這里簡(jiǎn)化為查出來(lái)
Long userId = jdbcTemplate.queryForObject("SELECT MAX(id) FROM users", Long.class);
jdbcTemplate.update(orderSql, productName, userId);
}
}4. 創(chuàng)建自定義注解(用于標(biāo)記需要事務(wù)的方法)
// TransactionalAspect.java
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface TransactionalAspect {
}5. 編寫切面:手動(dòng)控制事務(wù)
@Aspect
@Component
public class TransactionAspect {
@Autowired
private PlatformTransactionManager transactionManager;
@Around("@annotation(transactionalAspect)")
public Object manageTransaction(ProceedingJoinPoint joinPoint, TransactionalAspect transactionalAspect) throws Throwable {
// 1. 定義事務(wù)屬性
DefaultTransactionDefinition def = new DefaultTransactionDefinition();
def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED); // 默認(rèn)傳播行為
// 2. 開啟事務(wù)
TransactionStatus status = transactionManager.getTransaction(def);
try {
// 3. 執(zhí)行目標(biāo)方法
Object result = joinPoint.proceed();
// 4. 方法成功執(zhí)行,提交事務(wù)
transactionManager.commit(status);
System.out.println("事務(wù)提交成功");
return result;
} catch (Exception e) {
// 5. 發(fā)生異常,回滾事務(wù)
if (status != null && !status.isCompleted()) {
transactionManager.rollback(status);
System.out.println("事務(wù)已回滾,原因: " + e.getMessage());
}
throw e; // 重新拋出異常
}
}
}6. 測(cè)試 Controller
@RestController
public class TestController {
@Autowired
private UserService userService;
// 測(cè)試:成功場(chǎng)景
@GetMapping("/test/success")
public String testSuccess() {
userService.createUserWithOrder("張三", "iPhone");
return "用戶和訂單創(chuàng)建成功!";
}
// 測(cè)試:失敗場(chǎng)景(應(yīng)回滾)
@GetMapping("/test/fail")
public String testFail() {
try {
userService.createUserWithOrder("李四", "fail");
} catch (Exception e) {
return "創(chuàng)建失敗,但事務(wù)已回滾!";
}
return "success";
}
}四、測(cè)試驗(yàn)證
1. 啟動(dòng)應(yīng)用
訪問:
http://localhost:8080/test/success- 結(jié)果:用戶和訂單都插入
http://localhost:8080/test/fail- 結(jié)果:拋出異常,用戶也不會(huì)被保存(事務(wù)回滾)
? 驗(yàn)證成功:即使用戶先插入,但由于異常,整個(gè)事務(wù)回滾。
五、關(guān)鍵點(diǎn)解析
| 問題 | 解釋 |
|---|---|
為什么不用 @Transactional? | 演示 AOP 如何手動(dòng)控制事務(wù),理解底層原理 |
PlatformTransactionManager 從哪來(lái)? | Spring Boot 自動(dòng)配置(如 DataSourceTransactionManager) |
@Around 是關(guān)鍵 | 它能控制方法執(zhí)行前后,決定提交或回滾 |
| 異常必須 re-throw | 否則上層不知道失敗,可能掩蓋問題 |
status.isCompleted() | 防止重復(fù)提交或回滾 |
六、注意事項(xiàng)
- 切面失效問題:
- 不要在同一個(gè)類中調(diào)用被切面攔截的方法(內(nèi)部調(diào)用不走代理)
- 事務(wù)傳播行為:
- 當(dāng)前例子是
REQUIRED,更復(fù)雜的場(chǎng)景需考慮REQUIRES_NEW等
- 當(dāng)前例子是
- 性能:
- 事務(wù)越短越好,避免在事務(wù)中做耗時(shí)操作(如遠(yuǎn)程調(diào)用)
- 只讀事務(wù):
- 查詢方法應(yīng)使用
@Transactional(readOnly = true)
- 查詢方法應(yīng)使用
到此這篇關(guān)于Spring 中的切面與事務(wù)結(jié)合使用完整示例的文章就介紹到這了,更多相關(guān)Spring切面與事務(wù)結(jié)合內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java后臺(tái)判斷ajax請(qǐng)求及處理過程詳解
這篇文章主要介紹了Java后臺(tái)判斷ajax請(qǐng)求及處理過程詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-03-03
java實(shí)現(xiàn)簡(jiǎn)單貪吃蛇小游戲
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)單貪吃蛇小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-05-05
Java基于Log4j2實(shí)現(xiàn)異步日志系統(tǒng)的性能優(yōu)化實(shí)踐指南
在高并發(fā)的后端應(yīng)用中,日志記錄往往成為性能瓶頸之一,本文將從原理層面解析 Log4j2 異步Appender 與 Disruptor 工作機(jī)制,并結(jié)合 Spring Boot 業(yè)務(wù)場(chǎng)景給出最佳實(shí)踐配置與性能調(diào)優(yōu)建議2025-07-07
JAVA把結(jié)果保留兩位小數(shù)的3種方法舉例
在寫程序的時(shí)候,有時(shí)候可能需要設(shè)置小數(shù)的位數(shù),所以下面這篇文章主要給大家介紹了關(guān)于JAVA把結(jié)果保留兩位小數(shù)的3種方法,文章通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-08-08

