(轉)@Transactional spring 配置事務 注意事項

近日來,發現有很多童鞋詢問:
“Mybatis整合Spring3,事務注解為何不起作用?”
“已經聲明了事務,但是無法回滾。。。”
“Mybatis如果配置事務,Spring配置沒起作用啊!”
等等。。。

實際上,無論怎么問或者怎么貼出代碼,實際上沒有人能夠幫你解決這個問題的,首先Spring事務處理方式目前有五種,你用的到底是哪一種呢?回答問題的大神們不清楚,因此他們的回答和招在你那里不會起作用,因此,無論是哪一種事務處理方式首先你要弄明白你用的是哪一種,如果是雜交方式,建議選擇事務處理的第四種方式:使用tx標簽配置的攔截器,這個簡單而且容易上手,如果用這種方式還有問題,那請往下看,有很大可能性是由于這些原因造成的,我們在羅列代碼的時候一定要知其然還要知其所以然,這樣你不僅能夠快速的解決自己的問題,還能夠把該問題解決辦法與他人共享!

  1. 在需要事務管理的地方加@Transactional 注解。@Transactional 注解可以被應用于接口定義和接口方法、類定義和類的 public 方法上 。
  2. @Transactional 注解只能應用到 public 可見度的方法上 。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會報錯, 但是這個被注解的方法將不會展示已配置的事務設置。
  3. 注意僅僅 @Transactional 注解的出現不足于開啟事務行為,它僅僅 是一種元數據。必須在配置文件中使用配置元素,才真正開啟了事務行為。
  4. 通過 元素的 "proxy-target-class" 屬性值來控制是基于接口的還是基于類的代理被創 建。 如果 "proxy-target-class" 屬值被設置為 "true",那么基于類的代理將起作用(這時需要CGLIB庫cglib.jar在CLASSPATH中)。如果 "proxy-target-class" 屬值被設置為 "false" 或者這個屬性被省略,那么標準的JDK基于接口的代理將起作用。
<!-- JTA事務(非分布式事務), 事務配置的時候 ,不能指定dataSource屬性(分布式事務,是有全局事務來管理數據庫鏈接的)-->   
<!-- 標準的JDK基于接口的代理將起作用 -->  
<!-- aop切面 -->  
<aop:aspectj-autoproxy proxy-target-class="false" />  
  
<!-- 基于類的代理將起作用 ,同時 cglib.jar必須在CLASSPATH中 -->  
<!-- aop切面 -->  
<aop:aspectj-autoproxy proxy-target-class="true" />  

注 解@Transactional cglib與java動態代理最大區別是代理目標對象不用實現接口, 那么注解要是寫到接口方法上,要是使用cglib代理,這是注解事物就失效了,為了保持兼容注解最好都寫到實現類方法上。

  1. Spring團隊建議在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實現的任何接口上 。在接口上使用 @Transactional 注解,只能當你設置了基于接口的代理時它才生效。因為注解是 不能繼承 的,這就意味著如果正在使用基于類的代理時,那么事務的設置將不能被基于類的代理所識別,而且對象也將不會被事務代理所包裝。
  2. @Transactional 的事務開啟 ,或者是基于接口的 或者是基于類的代理被創建。所以在同一個類中一個方法調用另一個方法有事務的方法,事務是不會起作用的 。
public interface PersonageTempService {  
    //刪除指定id的Personage  
    public void del(Integer Personageid) ;  
  
    //刪除指定id的Personage,flag  
    public void del(Integer Personageid,boolean flag) ;  
}  
  
public class PersonageTempServiceBean implements PersonageTempService {  
    private JdbcTemplate jdbcTemplate;  
  
    public void del(Integer Personageid){  
        try{  
                this.del(Personageid,true)  
                System.out.println("del success");  
            }catch(Exception e){  
                System.out.println("del failed");  
            }  
        }  
  
        @Transactional  
        //此時,事務根本就沒有開啟, 即數據庫會默認提交該操作,即記錄別刪除掉  
        public void del(Integer Personageid,boolean flag){  
            if(flag == ture){  
                jdbcTemplate.update("del from Personage where id=?", new Object[]{Personageid}, new int[]{java.sql.Types.INTEGER});  
                throw new RuntimeException("運行期例外");  
            }  
        }  
    }  
  
    public class PersonageTempServiceBeanTest{  
        PersonageTempService ps = new PersonageTempServiceBean ();  
        ps.del(5);  
    }  
}
  1. Spring使用聲明式事務處理,默認情況下, 如果被注解的數據庫操作方法中發生了unchecked異常,所有的數據庫操作將rollback ;如果發生的異常是checked異常,默認情況下數 據庫操作還是會提 交的。
public interface PersonageService {  
    //刪除指定id的Personage  
    public void del(Integer Personageid) ;  
  
    //獲取Personage  
    public Personage getPersonage(Integer Personageid);  
}  
  
//PersonageServiceBean 實現了PersonageService 接口,則基于接口的還是基于類的代理 都可以實現事務  
@Transactional 
public class PersonageServiceBean implements PersonageService {  
    private JdbcTemplate jdbcTemplate;  
  
    //發生了unchecked異常,事務回滾, @Transactional  
    public void del(Integer Personageid){  
        jdbcTemplate.update("del from Personage where id=?", new Object[]{Personageid},  
        new int[]{java.sql.Types.INTEGER});  
        throw new RuntimeException("運行期例外");  
    }  
}  
public interface PersonageService {  
    //刪除指定id的Personage  
    public void delete(Integer Personageid) throws Exception;  
  
    //獲取Personage  
    public Personage getPersonage(Integer Personageid);  
}  
  
@Transactional  
public class PersonageServiceBean implements PersonageService {  
  
    //發生了checked異常,事務不回滾,即數據庫記錄仍能被刪除,  
    //checked的例外,需要我們在外部用try/catch語法對調用該方法的地方進行包含 
   @Transactional  
    public void delete(Integer Personageid) throws Exception{  
        jdbcTemplate.update("delete from Personage where id=?", new Object[]{Personageid},  
        new int[]{java.sql.Types.INTEGER});  
        throw new Exception("運行期例外");  
    }  
}  

但是,對于checked這種例外,默認情況下它是不會進行事務回滾的,但是 如果我們需要它進行事務回滾,這時候可以在delete方法上通過@Transaction這個注解來修改它的行為。

@Transactional  
public class PersonServiceBean implements PersonService {  
  
    @Transactional(rollbackFor=Exception.class)  
    //rollbackFor這屬性指定了,既使你出現了checked這種例外,那么它也會對事務進行回滾  
    public void delete(Integer personid) throws Exception{  
        jdbcTemplate.update("delete from person where id=?", new Object[]{personid},  
        new int[]{java.sql.Types.INTEGER});  
        throw new Exception("運行期例外");  
    }  
}  

在PersonServiceBean這個業務bean里面,有一些事務是不需要事務管理的,好比說獲取數據的getPersons方法,getPerson方法。因為@Transactional 放在了類的上面。

此時,可 以采用propagation這個事務屬性 @Transactional(propagation=Propagation.NOT_SUPPORTED),propagation這個屬性指定了 事務傳播行為,我們可以指定它不支持事務,當我們這么寫了之后,Spring容器在getPersons方法執行前就不會開啟事務 .

@Transactional  
public class PersonServiceBean implements PersonService {  
    @Transactional(propagation=Propagation.NOT_SUPPORTED)  
    //則此方法 就不會開啟事務了  
    public Person getPerson(Integer personid)  
    {  
    }  
}  
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • Spring Boot 參考指南 介紹 轉載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,951評論 6 342
  • 這部分的參考文檔涉及數據訪問和數據訪問層和業務或服務層之間的交互。 Spring的綜合事務管理支持覆蓋很多細節,然...
    竹天亮閱讀 1,049評論 0 0
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,915評論 18 139
  • 對大多數Java開發者來說,Spring事務管理是Spring應用中最常用的功能,使用也比較簡單。本文主要從三個方...
    sherlockyb閱讀 3,237評論 0 18
  • 殘雪初消薺滿園,糝羹珍美勝羔豚。吾曹舌本能知此,古學工夫始可言。
    XX像條狗閱讀 466評論 0 0