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.自我調(diào)用問題
當一個類中的方法調(diào)用同一個類中的另一個 @Transactional 方法時,事務不會傳播到被調(diào)用的方法。
示例代碼:
@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); } }
總結(jié)
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-06Spring?AOP?的實現(xiàn)和切點表達式的實現(xiàn)方式
本文給大家介紹了Spring?AOP的基本概念、通知類型、切點表達式和切面優(yōu)先級,并通過示例代碼展示了如何實現(xiàn)這些功能,感興趣的朋友跟隨小編一起看看吧2024-12-12java如何根據(jù)IP獲取當前區(qū)域天氣信息詳解
根據(jù)IP自動獲取當?shù)氐奶鞖忸A報信息這個功能大家應該都遇到過,天氣預報信息用途非常廣泛,篇文章主要給大家介紹了關于java如何根據(jù)IP獲取當前區(qū)域天氣信息的相關資料,需要的朋友可以參考下2021-08-08Spring中的@Transactional事務失效場景解讀
這篇文章主要介紹了Spring中的@Transactional事務失效場景解讀,如果Transactional注解應用在非public 修飾的方法上,Transactional將會失效此方法會檢查目標方法的修飾符是否為 public,不是 public則不會獲取@Transactional 的屬性配置信息,需要的朋友可以參考下2023-12-12