@Transactional 事務處理注解詳解
一、 事物注解方式: @Transactional
- 當标于類前時, 标示類中所有方法都進行事物處理 , 例子:
@Transactional
public class TestServiceBean implements TestService {}
- 當類中某些方法不需要事物時:
@Transactional
public class TestServiceBean implements TestService {
private TestDao dao;
public void setDao(TestDao dao) {
this.dao = dao;
}
@Transactional(propagation =Propagation.NOT_SUPPORTED)
public List getAll() {
return null;
}
}
- 或者不寫在類上方,單獨寫在某個方法之上
二、 事物傳播行為介紹:
- @Transactional(propagation=Propagation.REQUIRED) :如果有事務, 那麼加入事務, 沒有的話建立一個(預設情況下)
- @Transactional(propagation=Propagation.NOT_SUPPORTED) :容器不為這個方法開啟事務
- @Transactional(propagation=Propagation.REQUIRES_NEW) :不管是否存在事務,都建立一個新的事務,原來的挂起,新的執行完畢,繼續執行老的事務
- @Transactional(propagation=Propagation.MANDATORY) :必須在一個已有的事務中執行,否則抛出異常
- @Transactional(propagation=Propagation.NEVER) :必須在一個沒有的事務中執行,否則抛出異常(與Propagation.MANDATORY相反)
- @Transactional(propagation=Propagation.SUPPORTS) :如果其他bean調用這個方法,在其他bean中聲明事務,那就用事務.如果其他bean沒有聲明事務,那就不用事務.
- @Transactional(propagation=Propagation. NESTED):支援目前事務,如果目前事務存在,則執行一個嵌套事務,如果目前沒有事務,就建立一個事務。
三、 事物逾時設定:
@Transactional(timeout=30) //預設是30秒
四、 事務隔離級别:
- @Transactional(isolation = Isolation.READ_UNCOMMITTED):讀取未送出資料(會出現髒讀, 不可重複讀) 基本不使用
- @Transactional(isolation = Isolation.READ_COMMITTED):讀取已送出資料(會出現不可重複讀和幻讀)
- @Transactional(isolation = Isolation.REPEATABLE_READ):可重複讀(會出現幻讀)
- @Transactional(isolation = Isolation.SERIALIZABLE):串行化
MYSQL: 預設為REPEATABLE_READ級别
SQLSERVER: 預設為READ_COMMITTED
Oracle資料庫支援READ COMMITTED 和 SERIALIZABLE這兩種事務隔離級别。是以Oracle不支援髒讀
髒讀 : 一個事務讀取到另一事務未送出的更新資料
不可重複讀 : 在同一事務中, 多次讀取同一資料傳回的結果有所不同, 換句話說,
後續讀取可以讀到另一事務已送出的更新資料. 相反, "可重複讀"在同一事務中多次
讀取資料時, 能夠保證所讀資料一樣, 也就是後續讀取不能讀到另一事務已送出的更新資料
幻讀 : 一個事務讀到另一個事務已送出的insert資料
五、@Transactional注解中常用參數說明
參數名稱 | 功能描述 |
readOnly | 該屬性用于設定目前事務是否為隻讀事務,設定為true表示隻讀,false則表示可讀寫,預設值為false。例如:@Transactional(readOnly=true) |
rollbackFor | 該屬性用于設定需要進行復原的異常類數組,當方法中抛出指定異常數組中的異常時,則進行事務復原。例如: |
指定單一異常類:@Transactional(rollbackFor=RuntimeException.class) | |
指定多個異常類:@Transactional(rollbackFor={RuntimeException.class, Exception.class}) | |
rollbackForClassName | 該屬性用于設定需要進行復原的異常類名稱數組,當方法中抛出指定異常名稱數組中的異常時,則進行事務復原。例如: |
指定單一異常類名稱:@Transactional(rollbackForClassName="RuntimeException") | |
指定多個異常類名稱:@Transactional(rollbackForClassName={"RuntimeException","Exception"}) | |
noRollbackFor | 該屬性用于設定不需要進行復原的異常類數組,當方法中抛出指定異常數組中的異常時,不進行事務復原。例如: |
指定單一異常類:@Transactional(noRollbackFor=RuntimeException.class) | |
指定多個異常類:@Transactional(noRollbackFor={RuntimeException.class, Exception.class}) | |
noRollbackForClassName | 該屬性用于設定不需要進行復原的異常類名稱數組,當方法中抛出指定異常名稱數組中的異常時,不進行事務復原。例如: |
指定單一異常類名稱:@Transactional(noRollbackForClassName="RuntimeException") | |
指定多個異常類名稱: | |
@Transactional(noRollbackForClassName={"RuntimeException","Exception"}) | |
propagation | 該屬性用于設定事務的傳播行為,具體取值可參考第二部分事物傳播行為介紹 |
例如:@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true) | |
isolation | 該屬性用于設定底層資料庫的事務隔離級别,事務隔離級别用于處理多事務并發的情況,通常使用資料庫的預設隔離級别即可,基本不需要進行設定 |
timeout | 該屬性用于設定事務的逾時秒數,預設值為-1表示永不逾時 |
六、 手動復原
在需要手動進行復原的地方,加入以下代碼,可實作手動復原
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
七、注意的幾點:
1、@Transactional 隻能被應用到public方法上, 對于其它非public的方法,如果标記了@Transactional也不會報錯,但方法沒有事務功能.
2、用 spring 事務管理器,由spring來負責資料庫的打開,送出,復原.預設遇到運作期例外(throw new RuntimeException("注釋");)會復原,即遇到不受檢查(unchecked)的例外時復原;而遇到需要捕獲的例外(throw new Exception("注釋");)不會復原,即遇到受檢查的例外(就是非運作時抛出的異常,編譯器會檢查到的異常叫受檢查例外或說受檢查異常)時,需我們指定方式來讓事務復原要想所有異常都復原,要加上 @Transactional( rollbackFor={Exception.class,其它異常}) .如果讓unchecked例外不復原: @Transactional(notRollbackFor=RunTimeException.class),如下:
@Transactional(rollbackFor=Exception.class) //指定復原,遇到異常Exception時復原
public void methodName() {
throw new Exception("注釋");
}
@Transactional(noRollbackFor=Exception.class)//指定不復原,遇到運作期例外(throw new RuntimeException("注釋");)會復原
public ItimDaoImpl getItemDaoImpl() {
throw new RuntimeException("注釋");
}
3、@Transactional 注解應該隻被應用到 public 可見度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會報錯, 但是這個被注解的方法将不會展示已配置的事務設定。
4、@Transactional 注解可以被應用于接口定義和接口方法、類定義和類的 public 方法上。然而,請注意僅僅 @Transactional 注解的出現不足于開啟事務行為,它僅僅 是一種中繼資料,能夠被可以識别 @Transactional 注解和上述的配置适當的具有事務行為的beans所使用。上面的例子中,其實正是 元素的出現 開啟 了事務行為。
5、Spring團隊的建議是你在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實作的任何接口上。你當然可以在接口上使用 @Transactional 注解,但是這将隻能當你設定了基于接口的代理時它才生效。因為注解是不能繼承的,這就意味着如果你正在使用基于類的代理時,那麼事務的設定将不能被基于類的代理所識别,而且對象也将不會被事務代理所包裝(将被确認為嚴重的)。是以,請接受Spring團隊的建議并且在具體的類上使用 @Transactional 注解。