天天看點

Spring: @Transactional中的propagation屬性

在Spring的事務管理中,我們可以使用@Transactional這一annotation來對事務進行聲明式的設定。具體而言,就是在類或者方法前添加@Transactional并傳入屬性參數以擷取所需要的Transaction特性。Spring中的@Transactional有5個屬性:Propagation、Isolation、Rollback Rules、Timeout和Read-Only,其中Propagation屬性定義了Transaction的邊界 — 是否使用Transaction、在Transaction已存在的情況下如何表現等。

在service類前加上@Transactional,聲明這個service所有方法需要事務管理。每一個業務方法開始時都會打開一個事務。 

Spring預設情況下會對運作期例外(RunTimeException)進行事務復原。這個例外是unchecked 

如果遇到checked意外就不復原。 

如何改變預設規則: 

1 讓checked例外也復原:在整個方法前加上 @Transactional(rollbackFor=Exception.class) 

2 讓unchecked例外不復原: @Transactional(notRollbackFor=RunTimeException.class) 

3 不需要事務管理的(隻查詢的)方法:@Transactional(propagation=Propagation.NOT_SUPPORTED) 

在整個方法運作前就不會開啟事務 

       還可以加上:@Transactional(propagation=Propagation.NOT_SUPPORTED,readOnly=true),這樣就做成一個隻讀事務,可以提高效率。 

       各種屬性的意義: 

       REQUIRED:業務方法需要在一個容器裡運作。如果方法運作時,已經處在一個事務中,那麼加入到這個事務,否則自己建立一個新的事務。 

       NOT_SUPPORTED:聲明方法不需要事務。如果方法沒有關聯到一個事務,容器不會為他開啟事務,如果方法在一個事務中被調用,該事務會被挂起,調用結束後,原先的事務會恢複執行。 

       REQUIRESNEW:不管是否存在事務,該方法總彙為自己發起一個新的事務。如果方法已經運作在一個事務中,則原有事務挂起,新的事務被建立。 

       MANDATORY:該方法隻能在一個已經存在的事務中執行,業務方法不能發起自己的事務。如果在沒有事務的環境下被調用,容器抛出例外。 

       SUPPORTS:該方法在某個事務範圍内被調用,則方法成為該事務的一部分。如果方法在該事務範圍外被調用,該方法就在沒有事務的環境下執行。 

       NEVER:該方法絕對不能在事務範圍内執行。如果在就抛例外。隻有該方法沒有關聯到任何事務,才正常執行。 

       NESTED:如果一個活動的事務存在,則運作在一個嵌套的事務中。如果沒有活動事務,則按REQUIRED屬性執行。它使用了一個單獨的事務,這個事務擁有多個可以復原的儲存點。内部事務的復原不會對外部事務造成影響。它隻對DataSourceTransactionManager事務管理器起效。 

觀察以下兩個定義了@Transactional的方法,innerMethod()模拟了Transaction已經存在的情況,outMethod()則模拟了不存在已定義Transaction的情況:

@Transactional
public void outMethod() {
    exampleDAO.doSomething();
    innerMethod();
}

@Transactional
public void innerMethod() {
    exampleDAO.doElse();
}
           

對于這兩個方法,定義不同的Propagation屬性值所産生的效果如下(Propagation.NESTED的情況較為複雜,在此忽略): 

Propagation屬性 outMethod innerMethod
Propagation.MANDATORY .抛出異常 .在outMethod的Transaction中運作
Propagation.NEVER .不在Transaction中運作 .抛出異常
Propagation.NOT_SUPPORTED .不在Transaction中運作 .outMethod的Transaction暫停直至innerMethod執行完畢
Propagation.REQUIRED ( 預設值 ) .新開一個Transaction并在其中運作 .在outMethod的Transaction中運作
Propagation.REQUIRES_NEW .新開一個Transaction并在其中運作 .outMethod的Transaction暫停直至innerMethod中新開的Transaction執行完畢
Propagation.SUPPORTS .不在Transaction中運作 .在outMethod的Transaction中運作

@Transactional的代碼定義:

[html]  view plain copy

Spring: @Transactional中的propagation屬性
Spring: @Transactional中的propagation屬性
  1. @Target({ElementType.METHOD, ElementType.TYPE})  
  2. @Retention(RetentionPolicy.RUNTIME)  
  3. @Inherited  
  4. @Documented  
  5. public @interface Transactional {  
  6.     String value() default "";  
  7.     Propagation propagation() default Propagation.REQUIRED;  
  8.     Isolation isolation() default Isolation.DEFAULT;  
  9.     int timeout() default TransactionDefinition.TIMEOUT_DEFAULT;  
  10.     boolean readOnly() default false;  
  11.     Class<? extends Throwable>[] rollbackFor() default {};  
  12.     String[] rollbackForClassName() default {};  
  13.     Class<? extends Throwable>[] noRollbackFor() default {};  
  14.     String[] noRollbackForClassName() default {};  
  15. }  

  基于源代碼,我們可以發現在@Transactional,原來有這麼多的屬性可以進行配置,進而達到複雜應用控制的目的。

http://www.tuicool.com/articles/6rEjAv