SpringBoot@Transactional注解事務

一、特性

先來了解一下@Transactional注解事務的特性吧,可以更好排查問題

1、service類標簽(一般不建議在接口上)上添加@Transactional,可以將整個類納入spring事務管理,在每個業(yè)務方法執(zhí)行時都會開啟一個事務,不過這些事務采用相同的管理方式。

2、@Transactional 注解只能應用到 public 可見度的方法上。 如果應用在protected、private或者 package可見度的方法上,也不會報錯,不過事務設置不會起作用。

3、默認情況下,Spring會對unchecked異常進行事務回滾;如果是checked異常則不回滾。
辣么什么是checked異常,什么是unchecked異常

java里面將派生于Error或者RuntimeException(比如空指針,1/0)的異常稱為unchecked異常,其他繼承自java.lang.Exception得異常統稱為Checked Exception,如IOException、TimeoutException等

辣么再通俗一點:你寫代碼出現的空指針等異常,會被回滾,文件讀寫,網絡出問題,spring就沒法回滾了。然后我教大家怎么記這個,因為很多同學容易弄混,你寫代碼的時候有些IOException我們的編譯器是能夠檢測到的,說以叫checked異常,你寫代碼的時候空指針等死檢測不到的,所以叫unchecked異常。這樣是不是好記一些啦

4、只讀事務:
@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true)
只讀標志只在事務啟動時應用,否則即使配置也會被忽略。
啟動事務會增加線程開銷,數據庫因共享讀取而鎖定(具體跟數據庫類型和事務隔離級別有關)。通常情況下,僅是讀取數據時,不必設置只讀事務而增加額外的系統開銷。

二:事務傳播模式

Propagation枚舉了多種事務傳播模式,部分列舉如下:

  • 1、REQUIRED(默認模式):業(yè)務方法需要在一個容器里運行。如果方法運行時,已經處在一個事務中,那么加入到這個事務,否則自己新建一個新的事務。

  • 2、NOT_SUPPORTED:聲明方法不需要事務。如果方法沒有關聯到一個事務,容器不會為他開啟事務,如果方法在一個事務中被調用,該事務會被掛起,調用結束后,原先的事務會恢復執(zhí)行。

  • 3、REQUIRESNEW:不管是否存在事務,該方法總匯為自己發(fā)起一個新的事務。如果方法已經運行在一個事務中,則原有事務掛起,新的事務被創(chuàng)建。

  • 4、 MANDATORY:該方法只能在一個已經存在的事務中執(zhí)行,業(yè)務方法不能發(fā)起自己的事務。如果在沒有事務的環(huán)境下被調用,容器拋出例外。

  • 5、SUPPORTS:該方法在某個事務范圍內被調用,則方法成為該事務的一部分。如果方法在該事務范圍外被調用,該方法就在沒有事務的環(huán)境下執(zhí)行。

  • 6、NEVER:該方法絕對不能在事務范圍內執(zhí)行。如果在就拋例外。只有該方法沒有關聯到任何事務,才正常執(zhí)行。

  • 7、NESTED:如果一個活動的事務存在,則運行在一個嵌套的事務中。如果沒有活動事務,則按REQUIRED屬性執(zhí)行。它使用了一個單獨的事務,這個事務擁有多個可以回滾的保存點。內部事務的回滾不會對外部事務造成影響。它只對DataSourceTransactionManager事務管理器起效。

上面引用至事務傳播模式

二:解決Transactional注解不回滾

1、檢查你方法是不是public的
idea直接會給出提示Methods annotated with ‘@Transactional’ must be overridable ,原理很簡單,private修飾的方式,spring無法生成動態(tài)代理。

@Transactional
private void deleteUser() throws MyException{
    userMapper.deleteUserA();
    int i = 1/0;
    userMapper.deleteUserB();
}

2、你的異常類型是不是unchecked異常
異步雖然拋出了,但是拋出的是非RuntimeException類型的異常,依舊不會生效。

@Transactional
public void deleteUser() throws MyException{
    userMapper.deleteUserA();
    try {
        int i = 1 / 0;
        userMapper.deleteUserB();
    } catch (Exception e) {
        throw new MyException();
    }
}

如果我想check異常也想回滾怎么辦,注解上面寫明異常類型即可(指定了回滾異常類型為Exception)

@Transactional(rollbackFor=Exception.class) 

類似的還有norollbackFor,自定義不回滾的異常

3、數據庫引擎要支持事務,如果是MySQL,注意表要使用支持事務的引擎,比如innodb,如果是myisam,事務是不起作用的

4、是否開啟了對注解的解析

    <tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

事務傳播屬性設置錯誤
注意傳播屬性的設置,比如設置了:PROPAGATION_NOT_SUPPORIED(以非事務的方式執(zhí)行,如果當前有事務則把當前事務掛起)。

5、spring是否掃描到你這個包,如下是掃描到org.test下面的包

<context:component-scan base-package="org.test" ></context:component-scan>

6、檢查是不是同一個類中的方法調用(如a方法調用同一個類中的b方法)
如果先調用deleteUser(),那么deleteUserA()是不會回滾的,其原因就是@Transactional根本沒生成代理,如果直接調用deleteUser2()那么沒問題,deleteUserA()會回滾。

public void deleteUser() throws MyException{
    deleteUser2();
}

@Transactional
public void deleteUser2() throws MyException{
    userMapper.deleteUserA();
    int i = 1 / 0;
    userMapper.deleteUserB();
}

修改方式,把當前類自己注入一下調用即可。

@Service
public class UserService {
    @Autowired
    private UserMapper userMapper;
    //自己注入自己
    @Autowired
    UserService userService;

    public void deleteUser() throws MyException{
        userService.deleteUser2();
    }

    @Transactional
    public void deleteUser2() throws MyException{
        userMapper.deleteUserA();
        int i = 1 / 0;
        userMapper.deleteUserB();
    }
}

7、異常是不是被你catch住了
當異常被捕獲后,并且沒有再拋出,那么deleteUserA是不會回滾的。

@Transactional
public void deleteUser() {
    userMapper.deleteUserA();
    try {
        int i = 1 / 0;
        userMapper.deleteUserB();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

8、新開啟一個線程
如下的方式deleteUserA()也不會回滾,因為spring實現事務的原理是通過ThreadLocal把數據庫連接綁定到當前線程中,新開啟一個線程獲取到的連接就不是同一個了。

@Transactional
public void deleteUser() throws MyException{
    userMapper.deleteUserA();
    try {
        //休眠1秒,保證deleteUserA先執(zhí)行
        Thread.sleep(1000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
    new Thread(() -> {
        int i = 1/0;
        userMapper.deleteUserB();
    }).start();    
}