Spring AbstractRoutingDataSource 源碼分析

** 本文基于Spring 4.1.6 RELEASE源碼 **


先看看之前咱們在spring-mybatis.xml文件中配置的動態(tài)數(shù)據(jù)源昭躺,如下:

<bean id="dataSource" class="com.ricky.codelab.spring.ds.DynamicRoutingDataSource">
    <property name="targetDataSources">
        <map key-type="com.ricky.codelab.spring.ds.RouteStrategy">
            <entry key="slave1" value-ref="slave1DataSource"/>
            <entry key="slave2" value-ref="slave2DataSource"/>
        </map>
    </property>
    <!-- 默認(rèn)目標(biāo)數(shù)據(jù)源為主庫數(shù)據(jù)源 -->
    <property name="defaultTargetDataSource" ref="masterDataSource"/>
</bean>

DynamicRoutingDataSource 繼承自org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource類领炫,代碼如下:

package com.ricky.codelab.spring.ds;

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

/**
 * 動態(tài)數(shù)據(jù)源切換
 *
 * @author Ricky Fung
 * @create 2016-10-18 22:41
 */
public class DynamicRoutingDataSource extends AbstractRoutingDataSource {

    @Override
    protected Object determineCurrentLookupKey() {

        return DynamicRoutingContextHolder.getRouteStrategy();
    }
}

AbstractRoutingDataSource 類的定義,源碼如下:

package org.springframework.jdbc.datasource.lookup;

public abstract class AbstractRoutingDataSource extends AbstractDataSource implements InitializingBean {
    private Map<Object, Object> targetDataSources;
    private Object defaultTargetDataSource;
    private boolean lenientFallback = true;
    private DataSourceLookup dataSourceLookup = new JndiDataSourceLookup();
    private Map<Object, DataSource> resolvedDataSources;
    private DataSource resolvedDefaultDataSource;
}

AbstractRoutingDataSource 實現(xiàn)了InitializingBean接口似舵,Spring容器啟動時會回調(diào)其afterPropertiesSet()方法葱峡,AbstractRoutingDataSource afterPropertiesSet()如下:

public void afterPropertiesSet() {
    if(this.targetDataSources == null) {
        throw new IllegalArgumentException("Property \'targetDataSources\' is required");
    } else {
        this.resolvedDataSources = new HashMap(this.targetDataSources.size());
        Iterator var1 = this.targetDataSources.entrySet().iterator();

        while(var1.hasNext()) {
            Entry entry = (Entry)var1.next();
            Object lookupKey = this.resolveSpecifiedLookupKey(entry.getKey());
            DataSource dataSource = this.resolveSpecifiedDataSource(entry.getValue());
            this.resolvedDataSources.put(lookupKey, dataSource);
        }

        if(this.defaultTargetDataSource != null) {
            this.resolvedDefaultDataSource = this.resolveSpecifiedDataSource(this.defaultTargetDataSource);
        }

    }
}

protected DataSource resolveSpecifiedDataSource(Object dataSource) throws IllegalArgumentException {
    if(dataSource instanceof DataSource) {
        return (DataSource)dataSource;
    } else if(dataSource instanceof String) {
        return this.dataSourceLookup.getDataSource((String)dataSource);
    } else {
        throw new IllegalArgumentException("Illegal data source value - only [javax.sql.DataSource] and String supported: " + dataSource);
    }
}

這里主要把在xml中配置的targetDataSources解析到resolvedDataSources砰奕,defaultTargetDataSource 賦值到resolvedDefaultDataSource 提鸟。

另外仅淑,AbstractRoutingDataSource 繼承自AbstractDataSource,而AbstractDataSource 是DataSource的一個子類涯竟,如下:

package org.springframework.jdbc.datasource;

public abstract class AbstractDataSource implements DataSource {
    protected final Log logger = LogFactory.getLog(this.getClass());

    public AbstractDataSource() {
    }
}

因此,我們只需要看AbstractRoutingDataSource 的getConnection()方法實現(xiàn)即可蝗蛙,如下:

public Connection getConnection() throws SQLException {
    return this.determineTargetDataSource().getConnection();
}

public Connection getConnection(String username, String password) throws SQLException {
    return this.determineTargetDataSource().getConnection(username, password);
}

determineTargetDataSource()方法如下:


protected DataSource determineTargetDataSource() {
    Assert.notNull(this.resolvedDataSources, "DataSource router not initialized");
    Object lookupKey = this.determineCurrentLookupKey();
    DataSource dataSource = (DataSource)this.resolvedDataSources.get(lookupKey);
    if(dataSource == null && (this.lenientFallback || lookupKey == null)) {
        dataSource = this.resolvedDefaultDataSource;
    }

    if(dataSource == null) {
        throw new IllegalStateException("Cannot determine target DataSource for lookup key [" + lookupKey + "]");
    } else {
        return dataSource;
    }
}

determineTargetDataSource方法決定返回哪個DataSource 對象醉鳖,它根據(jù)determineCurrentLookupKey方法(這個方法正是我們在DynamicRoutingDataSource類中重寫的方法)獲取數(shù)據(jù)源的key哮内,然后從 resolvedDataSources中根據(jù)key取出DataSource ,如果返回的DataSource 為空纹因,則使用默認(rèn)的DataSource琳拨。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市狱庇,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌颜启,老刑警劉巖浪讳,帶你破解...
    沈念sama閱讀 206,839評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異淹遵,居然都是意外死亡,警方通過查閱死者的電腦和手機济炎,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評論 2 382
  • 文/潘曉璐 我一進店門淌实,熙熙樓的掌柜王于貴愁眉苦臉地迎上來猖腕,“玉大人恨闪,你說我怎么就攤上這事×剩” “怎么了?”我有些...
    開封第一講書人閱讀 153,116評論 0 344
  • 文/不壞的土叔 我叫張陵蜡豹,是天一觀的道長溉苛。 經(jīng)常有香客問我,道長娇唯,這世上最難降的妖魔是什么寂玲? 我笑而不...
    開封第一講書人閱讀 55,371評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮拓哟,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘流纹。我一直安慰自己逢倍,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 64,384評論 5 374
  • 文/花漫 我一把揭開白布较雕。 她就那樣靜靜地躺著,像睡著了一般扣典。 火紅的嫁衣襯著肌膚如雪慎玖。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,111評論 1 285
  • 那天湿硝,我揣著相機與錄音薪前,去河邊找鬼关斜。 笑死,一個胖子當(dāng)著我的面吹牛垛膝,可吹牛的內(nèi)容都是我干的丁稀。 我是一名探鬼主播,決...
    沈念sama閱讀 38,416評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼线衫,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了矿酵?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,053評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎棘捣,沒想到半個月后休建,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,558評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡测砂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,007評論 2 325
  • 正文 我和宋清朗相戀三年砌些,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片存璃。...
    茶點故事閱讀 38,117評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡纵东,死狀恐怖粘招,靈堂內(nèi)的尸體忽然破棺而出偎球,到底是詐尸還是另有隱情辑甜,我是刑警寧澤袍冷,帶...
    沈念sama閱讀 33,756評論 4 324
  • 正文 年R本政府宣布磷醋,位于F島的核電站,受9級特大地震影響难裆,放射性物質(zhì)發(fā)生泄漏子檀。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,324評論 3 307
  • 文/蒙蒙 一乃戈、第九天 我趴在偏房一處隱蔽的房頂上張望褂痰。 院中可真熱鬧,春花似錦症虑、人聲如沸缩歪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,315評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽匪蝙。三九已至,卻和暖如春习贫,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背苫昌。 一陣腳步聲響...
    開封第一講書人閱讀 31,539評論 1 262
  • 我被黑心中介騙來泰國打工颤绕, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人祟身。 一個月前我還...
    沈念sama閱讀 45,578評論 2 355
  • 正文 我出身青樓奥务,卻偏偏與公主長得像,于是被迫代替她去往敵國和親袜硫。 傳聞我的和親對象是個殘疾皇子氯葬,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,877評論 2 345

推薦閱讀更多精彩內(nèi)容

  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,748評論 6 342
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)婉陷,斷路器帚称,智...
    卡卡羅2017閱讀 134,600評論 18 139
  • application的配置屬性。 這些屬性是否生效取決于對應(yīng)的組件是否聲明為Spring應(yīng)用程序上下文里的Bea...
    新簽名閱讀 5,353評論 1 27
  • 暮色趕走了溫暖的陽光秽澳,霸占了天地世杀。煙灰色的云朵一塊一塊地彌散開來搶占了明亮的天空。又是一日要過去了肝集。雖是舍不得瞻坝,卻...
    菡萏琪玲閱讀 566評論 0 0
  • 不合適的人,分開該慶幸,因為終于不再互相折磨了所刀。 從這種意義上來講衙荐,沒跟那個最喜歡的人走到婚姻這一步,是一種幸運浮创,...
    云朵上的喵閱讀 1,401評論 0 3