程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> JAVA綜合教程 >> 項目中遇到的問題,項目遇到問題

項目中遇到的問題,項目遇到問題

編輯:JAVA綜合教程

項目中遇到的問題,項目遇到問題


一、Spring 事務問題

1.描述:service1 中的 a 調用 b,b 調用了 service2 中的 c ,c 調用了 service3 中的 d

期望:d 拋出異常時(我真實項目中拋出的是 Sql 異常),d,c 回滾,而 a,b 不回滾。

測試:考慮到 Spring 事務的自調用和 cglib 動態代理下的 spring 事務配置。添加了 <aop:aspectj-autoproxy expose-proxy="true" proxy-target-class="true"/>。

Demo:

自定義異常:

public class MyException extends SQLException { private static final long serialVersionUID = 1L; public MyException() { super(); } public MyException(String reason, String sqlState, int vendorCode, Throwable cause) { super(reason, sqlState, vendorCode, cause); } public MyException(String reason, String SQLState, int vendorCode) { super(reason, SQLState, vendorCode); } public MyException(String reason, String sqlState, Throwable cause) { super(reason, sqlState, cause); } public MyException(String reason, String SQLState) { super(reason, SQLState); } public MyException(String reason, Throwable cause) { super(reason, cause); } public MyException(String reason) { super(reason); } public MyException(Throwable cause) { super(cause); } } MyException.java

Dao:

@Repository public class TxDao { @Autowired private JdbcTemplate jdbcTemplate; public void updateA() { String sql = "update tx_test set a_field = 1 where id = 1"; jdbcTemplate.update(sql); } public void updateB() { String sql = "update tx_test set b_field = 1 where id = 1"; jdbcTemplate.update(sql); } public void updateC() { String sql = "update tx_test set c_field = 1 where id = 1"; jdbcTemplate.update(sql); } public void updateD() { String sql = "update tx_test set d_field = 1 where id = 1"; jdbcTemplate.update(sql); } } TxDao.java

ABService:

@Service public class ABService { @Autowired private TxDao txDao; @Autowired private CService cService; @Transactional public void aMethod() throws MyException { System.out.println("aMethod"); txDao.updateA(); ((ABService) AopContext.currentProxy()).bMetod(); } @Transactional public void bMetod() throws MyException { System.out.println("bMethod"); txDao.updateB(); cService.cMethod(); } } ABService.java

CService:

@Service public class CService { @Autowired private TxDao txDao; @Autowired private DService dService; @Transactional(rollbackFor=MyException.class) public void cMethod() throws MyException { System.out.println("cMethod..."); txDao.updateC(); dService.dMethod(); } } CService.java

DService:

@Service public class DService { @Autowired private TxDao txDao; @Transactional(rollbackFor=MyException.class) public void dMethod() throws MyException{ System.out.println("dMethod..."); txDao.updateD(); throw new MyException(); } } DService.java

測試代碼:

@Test public void test() { ABService service = context.getBean(ABService.class); try { service.aMethod(); } catch(MyException e) { e.printStackTrace(); } } View Code

(1)測試 rollbackFor 和 noRollbackFor

過程:

自定義了一個 Sql 異常 MyException 繼承自 SQLException,從 d 拋出,一直向上拋。

對 a, b 的 @Transactional 的 noRollbackFor 屬性設置為 MyException.class,而 c,d 的 rollbackFor 屬性設置為 MyException.class。

控制台輸出:

aMethod
bMethod
cMethod...
dMethod...

數據庫輸出:

測試結果: a,b,c,d 四個方法全部回滾。

原因查找:發現在容器初始化的時候就讀取了所有的 事務方法的 RollBackFor 和  noRollBackFor 屬性定義的 Value 值。

詳細參見:org.springframework.transaction.interceptor.RollbackRuleAttribute#RollbackRuleAttribute(java.lang.Class<?>)

在某個事務方法拋出異常後,整個事務都進行了回滾,感覺與配置 noRollBackFor 沒有關系。

詳細參見:org.springframework.transaction.interceptor.TransactionAspectSupport#invokeWithinTransaction

protected Object invokeWithinTransaction(Method method, Class<?> targetClass, final InvocationCallback invocation)
            throws Throwable {

// If the transaction attribute is null, the method is non-transactional.
final TransactionAttribute txAttr = getTransactionAttributeSource().getTransactionAttribute(method, targetClass);
final PlatformTransactionManager tm = determineTransactionManager(txAttr);
final String joinpointIdentification = methodIdentification(method, targetClass);

if (txAttr == null || !(tm instanceof CallbackPreferringPlatformTransactionManager)) {
    // Standard transaction demarcation with getTransaction and commit/rollback calls.
    TransactionInfo txInfo = createTransactionIfNecessary(tm, txAttr, joinpointIdentification);
    Object retVal = null;
    try {
        // This is an around advice: Invoke the next interceptor in the chain.
        // This will normally result in a target object being invoked.
        retVal = invocation.proceedWithInvocation();
    }
    catch (Throwable ex) {
        // target invocation exception
        completeTransactionAfterThrowing(txInfo, ex);
        throw ex;
    }
    finally {
        cleanupTransactionInfo(txInfo);
    }
    commitTransactionAfterReturning(txInfo);
    return retVal;
}

在標紅的地方會去掉目標方法,如目標方法拋出異常,則會進入到 catch 塊,執行完 catch 塊的代碼繼續向上拋出。catch 塊能捕獲到 MyException。

通過斷點跟蹤發現在 completeTransactionAfterThrowing() 方法裡進行了回滾,等回滾之後最後去調用了 commitTransactionAfterReturning() 方法。

這樣看來 noRollBackFor 甚至沒有什麼作用,有哪位大神看到這裡並且知道原理的話,請不吝賜教,謝謝。

(2)測試 Spring 事務的傳播行為。

過程:將 c 的傳播行為改為 REQUIRES_NEW,其他還是默認。同時在 c 方法中處理了 d 拋上來的異常。

控制台輸出:

aMethod
bMethod
cMethod...

數據庫輸出:

發現根本就不會去執行 d 方法。這裡就不是很明白了。

上面提供了兩個測試,事實上測試了  n 種方式,都沒有行的通。等以後對 spring 事務理解加深之後再來解析。這裡直接說最終的解決辦法。

解決辦法:

1.手動控制事務

2.服務拆分,比如這裡 a,b 單獨事務,c,d 單獨事務。

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved