您好,登錄后才能下訂單哦!
這篇文章主要介紹了Spring事務失效問題分析及解決方案,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
隔離級別
在 TransactionDefinition.java 接口中,定義了“四種”的隔離級別枚舉:
/** * 【Spring 獨有】使用后端數據庫默認的隔離級別 * * MySQL 默認采用的 REPEATABLE_READ隔離級別 * Oracle 默認采用的 READ_COMMITTED隔離級別 */ int ISOLATION_DEFAULT = -1; /** * 最低的隔離級別,允許讀取尚未提交的數據變更,可能會導致臟讀、幻讀或不可重復讀 */ int ISOLATION_READ_UNCOMMITTED = Connection.TRANSACTION_READ_UNCOMMITTED; /** * 允許讀取并發事務已經提交的數據,可以阻止臟讀,但是幻讀或不可重復讀仍有可能發生 */ int ISOLATION_READ_COMMITTED = Connection.TRANSACTION_READ_COMMITTED; /** * 對同一字段的多次讀取結果都是一致的,除非數據是被本身事務自己所修改,可以阻止臟讀和不可重復讀,但幻讀仍有可能發生。 */ int ISOLATION_REPEATABLE_READ = Connection.TRANSACTION_REPEATABLE_READ; /** * 最高的隔離級別,完全服從ACID的隔離級別。所有的事務依次逐個執行,這樣事務之間就完全不可能產生干擾,也就是說,該級別可以防止臟讀、不可重復讀以及幻讀。 * * 但是這將嚴重影響程序的性能。通常情況下也不會用到該級別。 */ int ISOLATION_SERIALIZABLE = Connection.TRANSACTION_SERIALIZABLE;
事務的傳播級別
事務的傳播行為,指的是當前帶有事務配置的方法,需要怎么處理事務;例如:方法可能繼續在現有事務中運行,也可能開啟一個新事務,并在自己的事務中運行;
需要注意,事務的傳播級別,并不是數據庫事務規范中的名詞,而是 Spring 自身所定義的。通過事務的傳播級別,Spring 才知道如何處理事務,是創建一個新事務呢,還是繼續使用當前的事務;
在 TransactionDefinition.java 接口中,定義了三類七種傳播級別:
// ========== 支持當前事務的情況 ========== /** * 如果當前存在事務,則使用該事務。 * 如果當前沒有事務,則創建一個新的事務。 */ int PROPAGATION_REQUIRED = 0; /** * 如果當前存在事務,則使用該事務。 * 如果當前沒有事務,則以非事務的方式繼續運行。 */ int PROPAGATION_SUPPORTS = 1; /** * 如果當前存在事務,則使用該事務。 * 如果當前沒有事務,則拋出異常。 */ int PROPAGATION_MANDATORY = 2; // ========== 不支持當前事務的情況 ========== /** * 創建一個新的事務。 * 如果當前存在事務,則把當前事務掛起。 */ int PROPAGATION_REQUIRES_NEW = 3; /** * 以非事務方式運行。 * 如果當前存在事務,則把當前事務掛起。 */ int PROPAGATION_NOT_SUPPORTED = 4; /** * 以非事務方式運行。 * 如果當前存在事務,則拋出異常。 */ int PROPAGATION_NEVER = 5; // ========== 其他情況 ========== /** * 如果當前存在事務,則創建一個事務作為當前事務的嵌套事務來運行。 * 如果當前沒有事務,則等價于 {@link TransactionDefinition#PROPAGATION_REQUIRED} */ int PROPAGATION_NESTED = 6;
@Transaction
@Transaction 注解是Spring的tx模塊提供的,使用AOP實現的事務控制,即底層為動態代理;
@Transaction 可以作用于接口、接口方法、類以及類方法上;當作用于類上時,該類的所有 public 方法將都具有該類型的事務屬性,也可以在方法級別使用該標注來覆蓋類級別的定義;
事務失效
1.異常類型錯誤,默認是RuntimException才會回滾
2.異常被catch后沒有拋出,需要拋異常才能回滾
3.是否發生自身調用的問題
4.注解所在位置是否為public修飾
5.數據源沒有配置事務管理器
6.不支持事務的引擎,如MyIsam
下面是一個事務失效的例子
@Slf4j @Service public class OrderServiceImpl implements OrderService { @Autowired private OrderMapper orderMapper; @Transactional @Override public void parent() { try { this.child(); } catch (Exception e) { log.error("插入異常", e); } Order order = new Order(); order.setOrderNo("parent"); order.setStatus("0"); order.setTitle("parent"); order.setAmount("1000"); orderMapper.insert(order); } @Transactional @Override public void child() { Order order = new Order(); order.setOrderNo("child"); order.setStatus("0"); order.setTitle("child"); order.setAmount("2000"); orderMapper.insert(order); throw new RuntimeException(); } }
當調用parent方法時,會調用child方法,但執行完成后插入的記錄有兩條,一條是parent的,一條是child的;
這是因為上面的parent方法調用的child方法出現問題,@Transaction 是基于AOP的方式進行事務控制的(CglibAopProxy.java進行方法攔截,TransactionInterceptor.java進行代理對象調用執行方法),需要使用的是代理對象調用方法,上面的代碼使用的還是this,即當前實例化對象,因此執行this.child(),方法不能被攔截增強;
將上面的parent方法修改如下
@Autowired private ApplicationContext context; private OrderService orderService; @PostConstruct public void init() { orderService = context.getBean(OrderService.class); } @Transactional @Override public void parent() { try { //獲取代理對象,通過代理對象調用child() orderService.child(); //獲取代理對象 //OrderService orderService = (OrderService) AopContext.currentProxy(); //orderService.child(); } catch (Exception e) { log.error("插入異常", e); } Order order = new Order(); order.setOrderNo("parent"); order.setStatus("0"); order.setTitle("parent"); order.setAmount("1000"); orderMapper.insert(order); }
執行parent方法,當出現異常的時候,事務會進行回滾;
如果想實現當調用parent方法時,調用child方法發生異常,只回滾child方法插入的數據,parent方法插入的數據不回滾,修改如下
@Transactional(propagation = Propagation.REQUIRES_NEW) @Override public void child() { Order order = new Order(); order.setOrderNo("child"); order.setStatus("0"); order.setTitle("child"); order.setAmount("2000"); orderMapper.insert(order); throw new RuntimeException(); }
@Transactional(propagation = Propagation.REQUIRES_NEW)
如果當前存在事務,則掛起事務并開啟一個新事務執行,新事務執行完畢后,喚醒之前的掛起的事務,則繼續執行;如果當前不存在事務,則新建一個事務;
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。