<context:component-scan base-package="com.zlkj" >
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller" />
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Service" />
</context:component-scan>
@Service("vehHpcjxSpService")
@Transactional
public class VehHpcjxSpServiceImpl implements VehHpcjxSpService{
@Override
@Transactional(propagation=Propagation.REQUIRED,rollbackFor={RuntimeException.class, Exception.class})
public ExcuteResult insertSpAndPhoto(VehHpcjxSp record,List<VehHpcjxPhoto> zps) throws RuntimeException{
ExcuteResult result = new ExcuteResult() ;
try{
result = vehHpcjxSpDao.insertSelective(record);
if(result.getReturnCode() == -1){
throw new RuntimeException("保存失敗");
}
if(zps!=null && zps.size()>0){
Iterator<VehHpcjxPhoto> it = zps.iterator();
while(it.hasNext()){
VehHpcjxPhoto zp = (VehHpcjxPhoto) it.next();
result = vehHpcjxPhotoDao.insert(zp);
if(result.getReturnCode() == -1){
throw new RuntimeException("保存失敗");
}
}
}
}catch(Exception ex){
throw new RuntimeException(ex.getMessage());
}
return result;
}
}
在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的情況:
補充: