@Transactional spring 配置事務 注意事項

@Transactional spring 配置事務 注意事項java

 

 

近日來,發現有不少童鞋詢問:「Mybatis整合Spring3,事務註解爲什麼不起做用? 」,「已經聲明瞭事務,可是沒法回滾。。。」「Mybatis若是配置事務,Spring配置沒起做用啊!」等等,實際上,不管怎麼問或者怎麼貼出代碼,實際上沒有人可以幫你解決這個問題的,首先Spring事務處理方式目前有五種,你用的究竟是哪種呢?回答問題的大神們不清楚,所以他們的回答和招在你那裏不會起做用,所以,不管是哪種事務處理方式首先你要弄明白你用的是哪種,若是是雜交方式,建議選擇事務處理的第四種方式:使用tx標籤配置的攔截器,這個簡單並且容易上手,若是用這種方式還有問題,那請往下看,有很大可能性是因爲這些緣由形成的,咱們在羅列代碼的時候必定要知其然還要知其因此然,這樣你不只可以快速的解決本身的問題,還可以把該問題解決辦法與他人共享!spring

 

 

1. 在須要事務管理的地方加@Transactional 註解。@Transactional 註解能夠被應用於接口定義和接口方法、類定義和類的 public 方法上 。sql

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基於接口的代理將起做用。this

Xml代碼  收藏代碼spa

  1. <!-- JTA事務(非分佈式事務), 事務配置的時候 ,不能指定dataSource屬性(分佈式事務,是有全局事務來管理數據庫連接的)-->   代理

  2. <!-- 標準的JDK基於接口的代理將起做用 -->  orm

  3. <!-- aop切面 -->  對象

  4.     <aop:aspectj-autoproxy proxy-target-class="false" />  

  5.   

  6. <!-- 基於類的代理將起做用 ,同時 cglib.jar必須在CLASSPATH中 -->  

  7. <!-- aop切面 -->  

  8.     <aop:aspectj-autoproxy proxy-target-class="true" />  

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

5. Spring團隊建議在具體的類(或類的方法)上使用 @Transactional 註解,而不要使用在類所要實現的任何接口上 。在接口上使用 @Transactional 註解,只能當你設置了基於接口的代理時它才生效。因爲註解是 不能繼承的,這就意味着若是正在使用基於類的代理時,那麼事務的設置將不能被基於類的代理所識別,並且對象也將不會被事務代理所包裝。

6. @Transactional 的事務開啓 ,或者是基於接口的 或者是基於類的代理被建立。因此在同一個類中一個方法調用另外一個方法有事務的方法,事務是不會起做用的 。

Java代碼  收藏代碼

  1. public interface PersonageTempService {  

  2.     //刪除指定id的Personage  

  3.     public void del(Integer Personageid) ;  

  4.   

  5.     //刪除指定id的Personage,flag  

  6.     public void del(Integer Personageid,boolean flag) ;  

  7.     }  

  8.   

  9.     public class PersonageTempServiceBean implements PersonageTempService {  

  10.         private JdbcTemplate jdbcTemplate;  

  11.   

  12.         public void del(Integer Personageid){  

  13.             try{  

  14.                 this.del(Personageid,true)  

  15.                 System.out.println("del success");  

  16.             }catch(Exception e){  

  17.                 System.out.println("del failed");  

  18.             }  

  19.         }  

  20.   

  21.         @Transactional  

  22.         //此時,事務根本就沒有開啓, 即數據庫會默認提交該操做,即記錄別刪除掉  

  23.         public void del(Integer Personageid,boolean flag){  

  24.             if(flag == ture){  

  25.                 jdbcTemplate.update("del from Personage where id=?"new Object[]{Personageid}, new int[]{java.sql.Types.INTEGER});  

  26.                 throw new RuntimeException("運行期例外");  

  27.             }  

  28.         }  

  29.     }  

  30.   

  31.     public class PersonageTempServiceBeanTest{  

  32.         PersonageTempService ps = new PersonageTempServiceBean ();  

  33.         ps.del(5);  

  34.     }  

  35. }  

 7. Spring使用聲明式事務處理,默認狀況下, 若是被註解的數據庫操做方法中發生了unchecked異常,全部的數據庫操做將rollback ;若是發生的異常是checked異常,默認狀況下數 據庫操做仍是會提 交的。

Java代碼  收藏代碼

  1. public interface PersonageService {  

  2.     //刪除指定id的Personage  

  3.     public void del(Integer Personageid) ;  

  4.   

  5.     //獲取Personage  

  6.     public Personage getPersonage(Integer Personageid);  

  7.     }  

  8.   

  9.     //PersonageServiceBean 實現了PersonageService 接口,則基於接口的仍是基於類的代理 均可以實現事務  

  10.     @Transactional public class PersonageServiceBean implements PersonageService {  

  11.     private JdbcTemplate jdbcTemplate;  

  12.   

  13.     //發生了unchecked異常,事務回滾, @Transactional  

  14.     public void del(Integer Personageid){  

  15.         jdbcTemplate.update("del from Personage where id=?"new Object[]{Personageid},  

  16.         new int[]{java.sql.Types.INTEGER});  

  17.         throw new RuntimeException("運行期例外");  

  18.     }  

  19. }  

 

Java代碼  收藏代碼

  1. public interface PersonageService {  

  2.     //刪除指定id的Personage  

  3.     public void delete(Integer Personageid) throws Exception;  

  4.   

  5.     //獲取Personage  

  6.     public Personage getPersonage(Integer Personageid);  

  7.     }  

  8.   

  9.     @Transactional  

  10.     public class PersonageServiceBean implements PersonageService {  

  11.   

  12.     //發生了checked異常,事務不回滾,即數據庫記錄仍能被刪除,  

  13.     //checked的例外,須要咱們在外部用try/catch語法對調用該方法的地方進行包含 @Transactional  

  14.     public void delete(Integer Personageid) throws Exception{  

  15.         jdbcTemplate.update("delete from Personage where id=?"new Object[]{Personageid},  

  16.         new int[]{java.sql.Types.INTEGER});  

  17.         throw new Exception("運行期例外");  

  18.     }  

  19. }  

 可是,對於checked這種例外,默認狀況下它是不會進行事務回滾的,可是 若是咱們須要它進行事務回滾,這時候能夠在delete方法上經過@Transaction這個註解來修改它的行爲。

Java代碼  收藏代碼

  1. @Transactional  

  2. public class PersonServiceBean implements PersonService {  

  3.   

  4.     @Transactional(rollbackFor=Exception.class)  

  5.     //rollbackFor這屬性指定了,既使你出現了checked這種例外,那麼它也會對事務進行回滾  

  6.     public void delete(Integer personid) throws Exception{  

  7.         jdbcTemplate.update("delete from person where id=?"new Object[]{personid},  

  8.         new int[]{java.sql.Types.INTEGER});  

  9.         throw new Exception("運行期例外");  

  10.     }  

  11. }  

 

在PersonServiceBean這個業務bean裏面,有一些事務是不須要事務管理的,比如說獲取數據的getPersons方法,getPerson方法。由於@Transactional 放在了類的上面。


此時,可 以採用propagation這個事務屬性 @Transactional(propagation=Propagation.NOT_SUPPORTED),propagation這個屬性指定了 事務傳播行爲,咱們能夠指定它不支持事務,當咱們這麼寫了以後,Spring容器在getPersons方法執行前就不會開啓事務 .

Java代碼  收藏代碼

  1. @Transactional  

  2. public class PersonServiceBean implements PersonService {  

  3.     @Transactional(propagation=Propagation.NOT_SUPPORTED)  

  4.     //則此方法 就不會開啓事務了  

  5.     public Person getPerson(Integer personid)  

  6.     {  

  7.     }  

  8. }  

相關文章
相關標籤/搜索