Spring常見的事務失效場景及解決方案
常見失效場景
1.方法不是 public
Spring AOP 代理默認只對 public 方法生效。如果被 @Transactional 注解的方法不是 public 的,事務將不會生效。
示例代碼
@Service
public class UserService {
@Transactional
protected void updateUser(User user) {
// 更新用戶信息
}
}
解決方案: 將方法改為 public。
@Service
public class UserService {
@Transactional
public void updateUser(User user) {
// 更新用戶信息
}
}
2.自我調用問題
當一個類中的方法調用同一個類中的另一個 @Transactional 方法時,事務不會傳播到被調用的方法。
示例代碼:
@Service
public class UserService {
@Transactional
public void updateUser(User user) {
internalUpdate(user);
}
@Transactional
private void internalUpdate(User user) {
// 更新用戶信息
}
}
解決方案: 使用 AopContext.currentProxy() 或者通過 @Autowired 注入當前類的代理對
@Service
public class UserService {
@Autowired
private UserService userService;
@Transactional
public void updateUser(User user) {
userService.internalUpdate(user);
}
@Transactional
private void internalUpdate(User user) {
// 更新用戶信息
}
}
3.異常被捕獲
如果在事務方法中捕獲了異常并且沒有重新拋出,Spring 框架無法檢測到異常,從而不會回滾事務。
示例代碼:
@Service
public class UserService {
@Transactional
public void updateUser(User user) {
try {
// 更新用戶信息
} catch (Exception e) {
// 異常被捕獲,事務不會回滾
}
}
}
解決方案: 捕獲異常后重新拋出,或者使用 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()。
@Service
public class UserService {
@Transactional
public void updateUser(User user) {
try {
// 更新用戶信息
} catch (Exception e) {
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}
}
}
4.配置錯誤
事務管理器配置錯誤或未正確配置,例如 @EnableTransactionManagement 缺失或事務管理器 bean 名稱不匹配。
示例代碼:
@Configuration
public class AppConfig {
@Bean
public PlatformTransactionManager transactionManager() {
return new DataSourceTransactionManager(dataSource());
}
@Bean
public DataSource dataSource() {
// 配置數(shù)據(jù)源
}
}
解決方案: 確保 @EnableTransactionManagement 注解存在,并且事務管理器配置正確。
@Configuration
@EnableTransactionManagement
public class AppConfig {
@Bean
public PlatformTransactionManager transactionManager() {
return new DataSourceTransactionManager(dataSource());
}
@Bean
public DataSource dataSource() {
// 配置數(shù)據(jù)源
}
}
5.事務傳播行為
不正確的事務傳播行為設置可能導致事務不按預期工作。例如,使用 REQUIRES_NEW 傳播行為時,新的事務會被創(chuàng)建,但原來的事務可能不會回滾。
示例代碼
@Service
public class UserService {
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void updateUser(User user) {
// 更新用戶信息
}
}
解決方案: 根據(jù)業(yè)務需求選擇合適的事務傳播行為。
@Service
public class UserService {
@Transactional(propagation = Propagation.REQUIRED)
public void updateUser(User user) {
// 更新用戶信息
}
}
6.事務隔離級別
不合適的事務隔離級別可能導致數(shù)據(jù)一致性問題,例如臟讀、不可重復讀和幻讀。
示例代碼
@Service
public class UserService {
@Transactional(isolation = Isolation.READ_UNCOMMITTED)
public void updateUser(User user) {
// 更新用戶信息
}
}
解決方案: 根據(jù)業(yè)務需求選擇合適的事務隔離級別。
@Service
public class UserService {
@Transactional(isolation = Isolation.READ_COMMITTED)
public void updateUser(User user) {
// 更新用戶信息
}
}
7.事務超時
事務超時設置不合理,導致事務在執(zhí)行過程中被自動回滾。
示例代碼:
@Service
public class UserService {
@Transactional(timeout = 1) // 超時時間設置為1秒
public void updateUser(User user) {
// 更新用戶信息
}
}
解決方案: 根據(jù)業(yè)務需求設置合理的事務超時時間。
@Service
public class UserService {
@Transactional(timeout = 60) // 超時時間設置為60秒
public void updateUser(User user) {
// 更新用戶信息
}
}
8.只讀事務
如果事務被標記為只讀(readOnly=true),則不能進行任何修改操作,否則會拋出異常。
示例代碼:
@Service
public class UserService {
@Transactional(readOnly = true)
public User getUserById(Long id) {
// 查詢用戶信息
}
}
解決方案: 確保只讀事務中不進行任何修改操作。
@Service
public class UserService {
@Transactional(readOnly = false)
public void updateUser(User user) {
// 更新用戶信息
}
}
9.數(shù)據(jù)庫連接問題
數(shù)據(jù)庫連接池配置不當或數(shù)據(jù)庫連接斷開,導致事務無法正常提交或回滾。
示例代碼:
@Configuration
public class AppConfig {
@Bean
public DataSource dataSource() {
HikariConfig config = new HikariConfig();
config.setJdbcUrl("jdbc:mysql://localhost:3306/test");
config.setUsername("root");
config.setPassword("password");
return new HikariDataSource(config);
}
}
解決方案: 確保數(shù)據(jù)庫連接池配置正確,并且數(shù)據(jù)庫連接穩(wěn)定。
@Configuration
public class AppConfig {
@Bean
public DataSource dataSource() {
HikariConfig config = new HikariConfig();
config.setJdbcUrl("jdbc:mysql://localhost:3306/test");
config.setUsername("root");
config.setPassword("password");
config.setMaximumPoolSize(10); // 設置最大連接數(shù)
return new HikariDataSource(config);
}
}
10.事務管理器不匹配
使用了錯誤的事務管理器,例如在使用 JPA 時配置了 JDBC 事務管理器,或者在使用 MyBatis 時配置了 Hibernate 事務管理器。
示例代碼:
@Configuration
public class AppConfig {
@Bean
public PlatformTransactionManager transactionManager(EntityManagerFactory emf) {
return new JpaTransactionManager(emf);
}
}
解決方案: 根據(jù)使用的持久層框架選擇合適的事務管理器。
@Configuration
public class AppConfig {
@Bean
public PlatformTransactionManager transactionManager(DataSource dataSource) {
return new DataSourceTransactionManager(dataSource);
}
}
總結
Spring 事務管理雖然強大,但在實際開發(fā)中需要注意許多細節(jié)。本文列舉了十種常見的事務失效情況,并提供了相應的解決方案和示例代碼。希望這些內(nèi)容能幫助開發(fā)者更好地理解和使用 Spring 事務管理,確保數(shù)據(jù)的一致性和完整性。
以上就是Spring常見的事務失效場景及解決方案的詳細內(nèi)容,更多關于Spring事務失效場景及解決的資料請關注腳本之家其它相關文章!
相關文章
如何使用HttpClient發(fā)送java對象到服務器
這篇文章主要介紹了如何使用HttpClient發(fā)送java對象到服務器,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-11-11
詳解在IDEA中使用MyBatis Generator逆向工程生成代碼
這篇文章主要介紹了詳解在IDEA中使用MyBatis Generator逆向工程生成代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-06-06
Spring?AOP?的實現(xiàn)和切點表達式的實現(xiàn)方式
本文給大家介紹了Spring?AOP的基本概念、通知類型、切點表達式和切面優(yōu)先級,并通過示例代碼展示了如何實現(xiàn)這些功能,感興趣的朋友跟隨小編一起看看吧2024-12-12
java如何根據(jù)IP獲取當前區(qū)域天氣信息詳解
根據(jù)IP自動獲取當?shù)氐奶鞖忸A報信息這個功能大家應該都遇到過,天氣預報信息用途非常廣泛,篇文章主要給大家介紹了關于java如何根據(jù)IP獲取當前區(qū)域天氣信息的相關資料,需要的朋友可以參考下2021-08-08
Spring中的@Transactional事務失效場景解讀
這篇文章主要介紹了Spring中的@Transactional事務失效場景解讀,如果Transactional注解應用在非public 修飾的方法上,Transactional將會失效此方法會檢查目標方法的修飾符是否為 public,不是 public則不會獲取@Transactional 的屬性配置信息,需要的朋友可以參考下2023-12-12

