<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    spring本地事務與JTA事務實現解析

    Posted on 2005-10-22 22:57 publisher luo 閱讀(10179) 評論(2)  編輯  收藏 所屬分類: 項目問題解決

          大家都知道spring支持兩種事務,一種是本地連接事務(使用DataSourceTransactionManager),一種是JTA事務(使用JtaTransactionManager)。并且支持聲明式事務和編程式事務兩種方式。采用聲明式事務使用AOP方式的TransactionProxyFactoryBean代理工廠類。
          JTA事務實現相對較好理解,在執行實際類的符合模式的方法時,代理類通過在連接點前后插入預處理過程(開始事務)和后處理過程(commit或rollbak)即可。因為JTA事務支持兩階段提交所以在代碼中啟動多少個連接(不同的connection)都能保證事務最終提交或者回滾。可是本地連接事務是如何實現的呢?因為必須后面的dao層必須使用的同一個連接才能保證事務正常提交和回滾,在業務邏輯層可以調用dao層的多個類的多個方法,每個方法如果顯式的將connection做為參數傳入到還可以,但是這樣connection就要出現調用的在業務邏輯層,而且dao的每個方法還要有個connection參數比較難看,而且開發人員要關注事務,這樣就沒有達到開發人員只要關注業務邏輯即可的要求。 
          web應用,各個類都要在多線程環境下運行,所以每個方法要保證線程安全,這樣,不在dao方法中加參數而是在dao類中加入connection屬性也就不可取了,怎么辦?查看一下JdbcTemplate類,在執行每個方法需要數據庫連接時都使用了DataSourceUtils.getConnection(getDataSource())方法?難道每回都從數據源里面取一條連接?不可能,這樣事務肯定沒法實現,可是怎么能保證取的是一條連接呢?對了是不是采用本地線程呀(TreadLocal),因為一段事務都是在一個線程中完成,所以只要在事務開始的時候將connection存放在本地線程中,然后事務過程中從本地線程中取出connection,直到事務結束即可。不錯,這樣只需要在每個dao方法的取數據庫連接的方法中有個事務狀態的判斷即可。不錯看看spring是不是這樣實現的?果然如此,DataSourceUtils.getConnection(DataSource)方法調用doGetConnection()方法,方法內容如下:
    public static Connection doGetConnection(DataSource dataSource) throws SQLException {
      Assert.notNull(dataSource, "No DataSource specified");

      ConnectionHolder conHolder = (ConnectionHolder) TransactionSynchronizationManager.getResource(dataSource);
      if (conHolder != null) {
       conHolder.requested();
       return conHolder.getConnection();
      }

      logger.debug("Fetching JDBC Connection from DataSource");
      Connection con = dataSource.getConnection();

      if (TransactionSynchronizationManager.isSynchronizationActive()) {
       logger.debug("Registering transaction synchronization for JDBC Connection");
       // Use same Connection for further JDBC actions within the transaction.
       // Thread-bound object will get removed by synchronization at transaction completion.
       conHolder = new ConnectionHolder(con);
       conHolder.setSynchronizedWithTransaction(true);
       conHolder.requested();
       TransactionSynchronizationManager.registerSynchronization(
         new ConnectionSynchronization(conHolder, dataSource));
       TransactionSynchronizationManager.bindResource(dataSource, conHolder);
      }

      return con;
     }
    conHolder?TransactionSynchronizationManager?很象呀,繼續看看TransactionSynchronizationManager類果真如此,里面使用TreadLocal來保存數據連接和事務狀態。原來如此,代碼里的各個層沒有特殊需要都不用再出現事務了,程序開發人員只要關注業務就可以了,不用再勞心編寫事務代碼了。

    Feedback

    # re: spring本地事務與JTA事務實現解析  回復  更多評論   

    2005-12-08 08:47 by seagoer
    不錯,我也一致疑惑 對于本地事務,難道spring實現了自己的jta,dao層每個方法都獨自獲得連接來處理,看起來好像確實是使用本地線程來處理的。

    # re: spring本地事務與JTA事務實現解析  回復  更多評論   

    2006-01-11 14:17 by barry
    呵呵。我也剛剛想到這個問題。
    查看了一下spring doc。
    如果在代碼中直接使用了DataSource.getConnetction()的方式(比如遺留代碼),還可以通過設置一個TransactionAwareDataSourceProxy來代理DataSource。

    posts - 9, comments - 27, trackbacks - 0, articles - 19

    Copyright © publisher luo

    主站蜘蛛池模板: 亚洲爆乳少妇无码激情| 国产精品无码免费专区午夜 | 最近中文字幕电影大全免费版 | 亚洲人成电影网站| 免费人成网站7777视频| 免费福利在线视频| 在线综合亚洲中文精品| 国产av无码专区亚洲av果冻传媒| 中文字幕亚洲免费无线观看日本| 亚洲Av永久无码精品一区二区| 亚洲中文字幕无码一久久区| 一个人免费观看视频www| 国产成人精品免费视频大全| 亚洲AV成人噜噜无码网站| 亚洲精品国产精品乱码不卡| 台湾一级毛片永久免费| 国产精品免费久久| 亚洲中文无码永久免费| 久久精品国产精品亚洲蜜月| 午夜无遮挡羞羞漫画免费| 久操视频免费观看| 人人爽人人爽人人片A免费 | 亚洲av午夜成人片精品网站 | 国产va免费精品观看精品| 成人网站免费大全日韩国产| 亚洲一区二区三区高清不卡| 国产亚洲精品资源在线26u| 日本人护士免费xxxx视频| 4hu四虎最新免费地址| 青青操免费在线视频| 日韩成人精品日本亚洲| 亚洲AV无码乱码在线观看代蜜桃| 亚洲AV无码久久精品成人| 又爽又高潮的BB视频免费看| 久久久久久99av无码免费网站| 一级毛片aaaaaa免费看| 99在线免费视频| 两性色午夜免费视频| 日韩毛片在线免费观看| 久久精品国产亚洲AV| 亚洲中文字幕乱码一区|