近日來,發(fā)現(xiàn)有很多童鞋詢問:
“Mybatis整合Spring3,事務(wù)注解為何不起作用?”
“已經(jīng)聲明了事務(wù),但是無法回滾。。?!?“Mybatis如果配置事務(wù),Spring配置沒起作用??!”
等等。。。
實(shí)際上,無論怎么問或者怎么貼出代碼,實(shí)際上沒有人能夠幫你解決這個(gè)問題的,首先Spring事務(wù)處理方式目前有五種,你用的到底是哪一種呢?回答問題的大神們不清楚,因此他們的回答和招在你那里不會(huì)起作用,因此,無論是哪一種事務(wù)處理方式首先你要弄明白你用的是哪一種,如果是雜交方式,建議選擇事務(wù)處理的第四種方式:使用tx標(biāo)簽配置的攔截器,這個(gè)簡單而且容易上手,如果用這種方式還有問題,那請(qǐng)往下看,有很大可能性是由于這些原因造成的,我們在羅列代碼的時(shí)候一定要知其然還要知其所以然,這樣你不僅能夠快速的解決自己的問題,還能夠把該問題解決辦法與他人共享!
- 在需要事務(wù)管理的地方加@Transactional 注解。@Transactional 注解可以被應(yīng)用于接口定義和接口方法、類定義和類的 public 方法上 。
- @Transactional 注解只能應(yīng)用到 public 可見度的方法上 。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不會(huì)報(bào)錯(cuò), 但是這個(gè)被注解的方法將不會(huì)展示已配置的事務(wù)設(shè)置。
- 注意僅僅 @Transactional 注解的出現(xiàn)不足于開啟事務(wù)行為,它僅僅 是一種元數(shù)據(jù)。必須在配置文件中使用配置元素,才真正開啟了事務(wù)行為。
- 通過 元素的 "proxy-target-class" 屬性值來控制是基于接口的還是基于類的代理被創(chuàng) 建。 如果 "proxy-target-class" 屬值被設(shè)置為 "true",那么基于類的代理將起作用(這時(shí)需要CGLIB庫cglib.jar在CLASSPATH中)。如果 "proxy-target-class" 屬值被設(shè)置為 "false" 或者這個(gè)屬性被省略,那么標(biāo)準(zhǔn)的JDK基于接口的代理將起作用。
<!-- JTA事務(wù)(非分布式事務(wù)), 事務(wù)配置的時(shí)候 ,不能指定dataSource屬性(分布式事務(wù),是有全局事務(wù)來管理數(shù)據(jù)庫鏈接的)-->
<!-- 標(biāo)準(zhǔn)的JDK基于接口的代理將起作用 -->
<!-- aop切面 -->
<aop:aspectj-autoproxy proxy-target-class="false" />
<!-- 基于類的代理將起作用 ,同時(shí) cglib.jar必須在CLASSPATH中 -->
<!-- aop切面 -->
<aop:aspectj-autoproxy proxy-target-class="true" />
注 解@Transactional cglib與java動(dòng)態(tài)代理最大區(qū)別是代理目標(biāo)對(duì)象不用實(shí)現(xiàn)接口, 那么注解要是寫到接口方法上,要是使用cglib代理,這是注解事物就失效了,為了保持兼容注解最好都寫到實(shí)現(xiàn)類方法上。
- Spring團(tuán)隊(duì)建議在具體的類(或類的方法)上使用 @Transactional 注解,而不要使用在類所要實(shí)現(xiàn)的任何接口上 。在接口上使用 @Transactional 注解,只能當(dāng)你設(shè)置了基于接口的代理時(shí)它才生效。因?yàn)樽⒔馐?不能繼承 的,這就意味著如果正在使用基于類的代理時(shí),那么事務(wù)的設(shè)置將不能被基于類的代理所識(shí)別,而且對(duì)象也將不會(huì)被事務(wù)代理所包裝。
- @Transactional 的事務(wù)開啟 ,或者是基于接口的 或者是基于類的代理被創(chuàng)建。所以在同一個(gè)類中一個(gè)方法調(diào)用另一個(gè)方法有事務(wù)的方法,事務(wù)是不會(huì)起作用的 。
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
//此時(shí),事務(wù)根本就沒有開啟, 即數(shù)據(jù)庫會(huì)默認(rèn)提交該操作,即記錄別刪除掉
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("運(yùn)行期例外");
}
}
}
public class PersonageTempServiceBeanTest{
PersonageTempService ps = new PersonageTempServiceBean ();
ps.del(5);
}
}
- Spring使用聲明式事務(wù)處理,默認(rèn)情況下, 如果被注解的數(shù)據(jù)庫操作方法中發(fā)生了unchecked異常,所有的數(shù)據(jù)庫操作將rollback ;如果發(fā)生的異常是checked異常,默認(rèn)情況下數(shù) 據(jù)庫操作還是會(huì)提 交的。
public interface PersonageService {
//刪除指定id的Personage
public void del(Integer Personageid) ;
//獲取Personage
public Personage getPersonage(Integer Personageid);
}
//PersonageServiceBean 實(shí)現(xiàn)了PersonageService 接口,則基于接口的還是基于類的代理 都可以實(shí)現(xiàn)事務(wù)
@Transactional
public class PersonageServiceBean implements PersonageService {
private JdbcTemplate jdbcTemplate;
//發(fā)生了unchecked異常,事務(wù)回滾, @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("運(yùn)行期例外");
}
}
public interface PersonageService {
//刪除指定id的Personage
public void delete(Integer Personageid) throws Exception;
//獲取Personage
public Personage getPersonage(Integer Personageid);
}
@Transactional
public class PersonageServiceBean implements PersonageService {
//發(fā)生了checked異常,事務(wù)不回滾,即數(shù)據(jù)庫記錄仍能被刪除,
//checked的例外,需要我們在外部用try/catch語法對(duì)調(diào)用該方法的地方進(jìn)行包含
@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("運(yùn)行期例外");
}
}
但是,對(duì)于checked這種例外,默認(rèn)情況下它是不會(huì)進(jìn)行事務(wù)回滾的,但是 如果我們需要它進(jìn)行事務(wù)回滾,這時(shí)候可以在delete方法上通過@Transaction這個(gè)注解來修改它的行為。
@Transactional
public class PersonServiceBean implements PersonService {
@Transactional(rollbackFor=Exception.class)
//rollbackFor這屬性指定了,既使你出現(xiàn)了checked這種例外,那么它也會(huì)對(duì)事務(wù)進(jìn)行回滾
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("運(yùn)行期例外");
}
}
在PersonServiceBean這個(gè)業(yè)務(wù)bean里面,有一些事務(wù)是不需要事務(wù)管理的,好比說獲取數(shù)據(jù)的getPersons方法,getPerson方法。因?yàn)锧Transactional 放在了類的上面。
此時(shí),可 以采用propagation這個(gè)事務(wù)屬性 @Transactional(propagation=Propagation.NOT_SUPPORTED),propagation這個(gè)屬性指定了 事務(wù)傳播行為,我們可以指定它不支持事務(wù),當(dāng)我們這么寫了之后,Spring容器在getPersons方法執(zhí)行前就不會(huì)開啟事務(wù) .
@Transactional
public class PersonServiceBean implements PersonService {
@Transactional(propagation=Propagation.NOT_SUPPORTED)
//則此方法 就不會(huì)開啟事務(wù)了
public Person getPerson(Integer personid)
{
}
}