欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

DynamicDataSource怎樣解決多數(shù)據(jù)源的事務問題

 更新時間:2023年07月31日 10:15:58   作者:Abstracted  
這篇文章主要介紹了DynamicDataSource怎樣解決多數(shù)據(jù)源的事務問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教

多數(shù)據(jù)源的真面目DynamicRoutingDataSource

DynamicRoutingDataSource是什么?

該類實現(xiàn)了DataSource接口并在內(nèi)部維護了一個map,其中存放了多個真實的DataSource,而key則是不同數(shù)據(jù)源的名稱,本質(zhì)上就是基于靜態(tài)代理模式代理了多個真實的數(shù)據(jù)源對象。

在用多數(shù)據(jù)源時可以使用@DS來切換不同的數(shù)據(jù)源,這依賴于DynamicRoutingDataSource

要完成這個功能需要

@DS+DSProcessor+ThreadLocal+AopMethodInterceptor+DynamicRoutingDataSource#getConnection

相互配合才能成功切換數(shù)據(jù)源。

多數(shù)據(jù)源下的事務會有什么問題?

在沒有多數(shù)據(jù)源時,系統(tǒng)中一般只會存在一個Datasource,所謂的HikariDataSource,DruidDatasource,CommonsDbcp2PoolDataSource都是DataSource的實現(xiàn),但是在系統(tǒng)中總是唯一存在的。因為SqlSessionTemplateFactory中只能存在一個DataSource,SqlSessionTemplate又是通過SqlSessionTemplateFactory生成的,并且SqlSessionTemplate在容器中又只能存在一個,所以事務時不會出現(xiàn)混亂的。而DynamicRoutingDataSource同樣也是系統(tǒng)中的唯一的Datasource,只不過它內(nèi)部代理了多個DataSource。

再來說說spring的事務管理器TransactionSynchronizationManager,內(nèi)部使用ThreadLocal保存當前事務信息。當程序執(zhí)行到@Transactional的AOP攔截器時,會在ThreadLocal中保存當前的事務信息,其中就包含了與數(shù)據(jù)庫的Connection對象。在程序執(zhí)行sql時,spring的SqlSessionTemplate#getSession方法會從事務管理器中獲取到與當前線程綁定的connection對象。

綜上所述,當使用@DS切換數(shù)據(jù)源時,沒有事務的情況下還好,會使用DynamicRoutingDataSource獲取一個新的connection并使用,但是如果是在事務的情況下,會使用事務管理器中獲取與當前線程綁定的Connection,而這個connection則是事務被創(chuàng)建時獲取的connection,就造成了雖然指定了數(shù)據(jù)源,但是還是原本的那個connection。導致切換數(shù)據(jù)源失敗。

@Ds與@DsTransactional

通過觀察DynamicDataSourceAutoConfiguration自動配置類可以發(fā)現(xiàn),DynamicDataSource默認自動配置了@Ds注解@DsTransactional注解的切面。

分別是dynamicDatasourceAnnotationAdvisordynamicTransactionAdvisor。

@Ds的原理

其切面方法攔截器會將指定的ds名稱存入DynamicDataSourceContextHolder中的ThreadLocal<Deque<String>>中,因為指定了ThreadLocalMap的泛型為Deque,而Deque的作用更準確的說是棧的作用,所以支持方法嵌套調(diào)用時使用不同的ds名稱。

@DsTransactional的原理

通過源碼可以發(fā)現(xiàn)在沒有使用seta分布式事務控制的情況下,多數(shù)據(jù)源的事務是通過dynamicDatasourceAnnotationAdvisor管理的。

dynamicDatasourceAnnotationAdvisor的核心切面攔截器就是DynamicLocalTransactionInterceptor

DynamicLocalTransactionInterceptor多數(shù)據(jù)源本地事務攔截器

內(nèi)部實現(xiàn)原理非常簡單,其實就是借助于ThreadLocal。

看到這里,你可能已經(jīng)猜到了,面紗下面的真面目就是這個ConnectionFactory類了。

我們再來看看他張了一幅什么面貌。

我們已經(jīng)知道了notify方法是@DsTransactional切面環(huán)繞通知結(jié)束時會被調(diào)用的,本著追溯本源的好奇心,你可能開始好奇了,putConnection又是什么時候被調(diào)用的呢?

我們繼續(xù)跟進就來到了AbstractRoutingDataSource。

好!又回到了DynamicRoutingDataSource中, AbstractRoutingDataSource就是DynamicRoutingDataSource父類。

在本地事務結(jié)束時,TransactionContext會清空本地事物的狀態(tài)標識,然后分別結(jié)束每一個connection的事務狀態(tài)。

然后清除ConnectionFactory中保存的與本次本地事物有關(guān)的所有connection對象的引用。

至此本地事物的創(chuàng)建和結(jié)束就完成了閉環(huán)。

總結(jié)一下

多數(shù)據(jù)源事務的控制中,參與的核心職責類有哪些

  • DynamicRoutingDataSource: DataSource接口的實現(xiàn),也是一個DataSource,本質(zhì)是多個DataSource的靜態(tài)代理類。自定義了getConnection方法的實現(xiàn)邏輯,使其與多數(shù)據(jù)源的事務管理緊密配合。
  • TransactionContext:使用ThreadLocal實現(xiàn)。本地事物上下文。負責管理本地事物的狀態(tài)。
  • ConnectionFactory: 使用ThreadLocal實現(xiàn)。connection連接的靜態(tài)代理類,也是一個委派者設計模式。負責管理當前本地事務中的所有Connection。
  • ConnectionProxy: 封裝了真實的Connection。 目的是將ds數(shù)據(jù)源名稱和connection對應起來,這樣可以通過ds數(shù)據(jù)源名稱拿到對應的connection。
  • @Ds:用于指定接下來要使用的數(shù)據(jù)源名稱。
  • @DsTransactional: 多數(shù)據(jù)源事務的開啟注解,用法同@Transactional相同。但是不能指定事務的一些屬性,因為其實現(xiàn)的原理,也不需要任何其他的事務的配置。當發(fā)生任何Exception時都會執(zhí)行回滾

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Java中Properties配置類用法詳解

    Java中Properties配置類用法詳解

    所謂的配置文件問題,是指我們在開發(fā)時,經(jīng)常需要讀取和修改一些配置信息,比如數(shù)據(jù)庫、消息隊列、Nginx、Web服務器等的配置,為了便于修改這些信息,我們可以采用Properties配置類,本文給大家講一下Properties配置類是怎么回事,以及怎么使用
    2023-06-06
  • Java全面細致講解類與對象

    Java全面細致講解類與對象

    類和對象是兩種以計算機為載體的計算機語言的合稱。對象是對客觀事物的抽象,類是對對象的抽象。類是一種抽象的數(shù)據(jù)類型;變量就是可以變化的量,存儲在內(nèi)存中—個可以擁有在某個范圍內(nèi)的可變存儲區(qū)域
    2022-05-05
  • idea 多模塊項目依賴父工程class找不到問題的方法

    idea 多模塊項目依賴父工程class找不到問題的方法

    這篇文章主要介紹了idea 多模塊項目依賴父工程class找不到問題的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2019-01-01
  • JAVA maven項目使用釘釘SDK獲取token、用戶

    JAVA maven項目使用釘釘SDK獲取token、用戶

    這篇文章主要介紹了JAVA maven項目使用釘釘SDK獲取token、用戶,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-06-06
  • Java如何實現(xiàn)數(shù)據(jù)壓縮所有方式性能測試

    Java如何實現(xiàn)數(shù)據(jù)壓縮所有方式性能測試

    本文介紹了多種壓縮算法及其在Java中的實現(xiàn),包括LZ4、BZip2、Deflate、Gzip和7z等,LZ4以其高效的壓縮和解壓縮速度而受到青睞,特別是在大數(shù)據(jù)處理場景中,通過對比不同壓縮算法的性能和壓縮率,我們選擇了最適合當前項目需求的壓縮工具
    2025-02-02
  • SpringCloud可視化鏈路追蹤系統(tǒng)Zipkin部署過程

    SpringCloud可視化鏈路追蹤系統(tǒng)Zipkin部署過程

    這篇文章主要介紹了SpringCloud可視化鏈路追蹤系統(tǒng)Zipkin部署過程,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2020-03-03
  • 深入探究SpringBoot中的Sleuth用法

    深入探究SpringBoot中的Sleuth用法

    Sleuth是一個分布式跟蹤系統(tǒng),用于跟蹤應用程序中的請求和操作,在本文中,我們將探討SpringBoot中的Sleuth是什么,以及如何使用它來跟蹤應用程序中的請求和操作,感興趣的小伙伴跟著小編一起來探討吧
    2023-07-07
  • Java使用Redis的方法實例分析

    Java使用Redis的方法實例分析

    這篇文章主要介紹了Java使用Redis的方法,接合實例形式分析了相關(guān)redis驅(qū)動包安裝、java連接redis服務器、數(shù)據(jù)存儲、讀取等相關(guān)操作技巧,需要的朋友可以參考下
    2018-05-05
  • Java多線程的其他知識_動力節(jié)點Java學院整理

    Java多線程的其他知識_動力節(jié)點Java學院整理

    這篇文章主要介紹了Java多線程的其他知識,需要的朋友可以參考下
    2017-05-05
  • Java實現(xiàn)動態(tài)IP代理的步驟詳解

    Java實現(xiàn)動態(tài)IP代理的步驟詳解

    在網(wǎng)絡編程中,動態(tài)IP代理可以幫助用戶隱藏真實IP以及提高數(shù)據(jù)抓取的效率,本文將介紹如何在Java中實現(xiàn)動態(tài)IP代理,包括設置代理、發(fā)送請求以及處理響應,需要的朋友可以參考下
    2025-02-02

最新評論