SpringBoot基于SpringAop實(shí)現(xiàn)Mysql的讀寫分離(多數(shù)據(jù)源)

本文源代碼主要來源于簡書作者“險(xiǎn)遠(yuǎn)的奇?zhèn)ピ幑帧钡奈恼?a href="http://www.reibang.com/p/8813ec02926a" target="_blank">Spring Boot 集成Mybatis實(shí)現(xiàn)主從(多數(shù)據(jù)源)分離方案案腺,感謝作者的奉獻(xiàn)枉疼,但是由于作者寫的代碼pull下來后未能實(shí)現(xiàn)主從分離垮卓,而且有諸多bug存在,遂寫該文記錄一下。


如下圖原作者評(píng)論區(qū),大部分讀者都未實(shí)現(xiàn)主從分離:

原作者的評(píng)論區(qū)

通過debug發(fā)現(xiàn)误证,啟動(dòng)后程序并沒有進(jìn)入MybatisConfiguration中的sqlSessionFactory的方法,所有主從配置未初始化到spring容器中修壕。究其原因愈捅,是因?yàn)樽髡叨x的方法和父類的方法名返回值都一模一樣,程序走了父類MybatisAutoConfiguration中sqlSessionFactory的方法默認(rèn)的配置慈鸠,所有從庫未生效蓝谨。

@Bean
  @ConditionalOnMissingBean
  public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
    SqlSessionFactoryBean factory = new SqlSessionFactoryBean();
    factory.setDataSource(dataSource);
    factory.setVfs(SpringBootVFS.class);
    if (StringUtils.hasText(this.properties.getConfigLocation())) {
      factory.setConfigLocation(this.resourceLoader.getResource(this.properties.getConfigLocation()));
    }
    factory.setConfiguration(properties.getConfiguration());
    if (!ObjectUtils.isEmpty(this.interceptors)) {
      factory.setPlugins(this.interceptors);
    }
    if (this.databaseIdProvider != null) {
      factory.setDatabaseIdProvider(this.databaseIdProvider);
    }
    if (StringUtils.hasLength(this.properties.getTypeAliasesPackage())) {
      factory.setTypeAliasesPackage(this.properties.getTypeAliasesPackage());
    }
    if (StringUtils.hasLength(this.properties.getTypeHandlersPackage())) {
      factory.setTypeHandlersPackage(this.properties.getTypeHandlersPackage());
    }
    if (!ObjectUtils.isEmpty(this.properties.resolveMapperLocations())) {
      factory.setMapperLocations(this.properties.resolveMapperLocations());
    }

    return factory.getObject();
  }

找到問題后,修改如下

@Configuration
@AutoConfigureAfter({ DataSourceConfiguration.class })
public class MybatisConfiguration extends MybatisAutoConfiguration {

    private static Log logger = LogFactory.getLog(MybatisConfiguration.class);

    @Resource(name = "masterDataSource")
    private DataSource masterDataSource;
    @Resource(name = "slaveDataSource")
    private DataSource slaveDataSource;

    
    @Bean
    public SqlSessionFactory sqlSessionFactoryTest() throws Exception {
        return super.sqlSessionFactory(roundRobinDataSouceProxy());
    }

    public ReadWriteSplitRoutingDataSource roundRobinDataSouceProxy() {
        ReadWriteSplitRoutingDataSource proxy = new ReadWriteSplitRoutingDataSource();
        Map<Object, Object> targetDataResources = new ClassLoaderRepository.SoftHashMap();
        // Map<Object, Object> targetDataResources = new HashMap<Object, Object>();
        targetDataResources.put(ReadWriteSplitRoutingDataSource.DbType.MASTER, masterDataSource);
        targetDataResources.put(ReadWriteSplitRoutingDataSource.DbType.SLAVE, slaveDataSource);
        proxy.setDefaultTargetDataSource(masterDataSource);
        proxy.setTargetDataSources(targetDataResources);

        return proxy;
    }
}

啟動(dòng)不報(bào)錯(cuò)了,但是訪問http://localhost:8080/dictype/all譬巫,報(bào)下面異常

java.lang.IllegalArgumentException: DataSource router not initialized

dataSource未初始化咖楣,debug發(fā)現(xiàn),源碼類AbstractRoutingDataSource中初始化DataSource發(fā)現(xiàn)芦昔,傳遞給父類的datasource需要是DataSource類型或者String類型诱贿,而我們?cè)谧远x類MybatisConfiguration中封裝DataSource的時(shí)候使用的是SoftHashMap

public static class SoftHashMap extends AbstractMap {
        private Map<Object, SpecialValue> map;
        boolean recordMiss = true; // only interested in recording miss stats sometimes
        private ReferenceQueue rq = new ReferenceQueue();

        public SoftHashMap(Map<Object, SpecialValue> map) {
            this.map = map;
        }

        public SoftHashMap() {
            this(new HashMap());
        }

觀察SoftHashMap源碼發(fā)現(xiàn):SoftHashMap的value值類型為SpecialValue類型,不是上面源碼需要的DataSource類型或者String類型咕缎,所以修改如下:

@Configuration
@AutoConfigureAfter({ DataSourceConfiguration.class })
public class MybatisConfiguration extends MybatisAutoConfiguration {

    private static Log logger = LogFactory.getLog(MybatisConfiguration.class);

    @Resource(name = "masterDataSource")
    private DataSource masterDataSource;
    @Resource(name = "slaveDataSource")
    private DataSource slaveDataSource;

    @Bean
    public SqlSessionFactory sqlSessionFactoryTest(ReadWriteSplitRoutingDataSource dataSource) throws Exception {
        logger.info("-------------------- 重載父類 sqlSessionFactory init---------------------");
        return super.sqlSessionFactory(dataSource);
    }

    @Bean
    public ReadWriteSplitRoutingDataSource roundRobinDataSouceProxy() {
        ReadWriteSplitRoutingDataSource proxy = new ReadWriteSplitRoutingDataSource();
        Map<Object, Object> targetDataResources = new HashMap<Object, Object>();
        targetDataResources.put(ReadWriteSplitRoutingDataSource.DbType.MASTER, masterDataSource);
        targetDataResources.put(ReadWriteSplitRoutingDataSource.DbType.SLAVE, slaveDataSource);
        proxy.setDefaultTargetDataSource(masterDataSource);
        proxy.setTargetDataSources(targetDataResources);

        return proxy;
    }
}

重啟后再次訪問http://localhost:8080/dictype/all珠十,發(fā)現(xiàn)完美的調(diào)用了從庫的地址:

init datasource error, url: jdbc:mysql://192.168.249.128:3381/db-test?characterEncoding=UTF-8&autoReconnect=true&zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=utf-8

com.mysql.jdbc.exceptions.jdbc4.MySQLNonTransientConnectionException: Could not create connection to database server. Attempted reconnect 3 times. Giving up.

至此,完美實(shí)現(xiàn)多數(shù)據(jù)源主從分離凭豪。

綜上焙蹭,遇到問題要多debug,多方位考慮問題的癥結(jié)所在。

修改后的源碼地址

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末嫂伞,一起剝皮案震驚了整個(gè)濱河市孔厉,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌帖努,老刑警劉巖撰豺,帶你破解...
    沈念sama閱讀 222,183評(píng)論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異然磷,居然都是意外死亡郑趁,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,850評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門姿搜,熙熙樓的掌柜王于貴愁眉苦臉地迎上來寡润,“玉大人,你說我怎么就攤上這事舅柜∷笪疲” “怎么了?”我有些...
    開封第一講書人閱讀 168,766評(píng)論 0 361
  • 文/不壞的土叔 我叫張陵致份,是天一觀的道長变抽。 經(jīng)常有香客問我,道長氮块,這世上最難降的妖魔是什么绍载? 我笑而不...
    開封第一講書人閱讀 59,854評(píng)論 1 299
  • 正文 為了忘掉前任,我火速辦了婚禮滔蝉,結(jié)果婚禮上击儡,老公的妹妹穿的比我還像新娘。我一直安慰自己蝠引,他們只是感情好阳谍,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,871評(píng)論 6 398
  • 文/花漫 我一把揭開白布蛀柴。 她就那樣靜靜地躺著,像睡著了一般矫夯。 火紅的嫁衣襯著肌膚如雪鸽疾。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,457評(píng)論 1 311
  • 那天训貌,我揣著相機(jī)與錄音制肮,去河邊找鬼。 笑死旺订,一個(gè)胖子當(dāng)著我的面吹牛弄企,可吹牛的內(nèi)容都是我干的超燃。 我是一名探鬼主播区拳,決...
    沈念sama閱讀 40,999評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼意乓!你這毒婦竟也來了樱调?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,914評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤届良,失蹤者是張志新(化名)和其女友劉穎笆凌,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體士葫,經(jīng)...
    沈念sama閱讀 46,465評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡乞而,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,543評(píng)論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了慢显。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片爪模。...
    茶點(diǎn)故事閱讀 40,675評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖荚藻,靈堂內(nèi)的尸體忽然破棺而出屋灌,到底是詐尸還是另有隱情,我是刑警寧澤应狱,帶...
    沈念sama閱讀 36,354評(píng)論 5 351
  • 正文 年R本政府宣布共郭,位于F島的核電站,受9級(jí)特大地震影響疾呻,放射性物質(zhì)發(fā)生泄漏除嘹。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,029評(píng)論 3 335
  • 文/蒙蒙 一岸蜗、第九天 我趴在偏房一處隱蔽的房頂上張望尉咕。 院中可真熱鬧,春花似錦散吵、人聲如沸龙考。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,514評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽晦款。三九已至炎功,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間缓溅,已是汗流浹背蛇损。 一陣腳步聲響...
    開封第一講書人閱讀 33,616評(píng)論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留坛怪,地道東北人淤齐。 一個(gè)月前我還...
    沈念sama閱讀 49,091評(píng)論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像袜匿,于是被迫代替她去往敵國和親更啄。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,685評(píng)論 2 360

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

  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,859評(píng)論 6 342
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理居灯,服務(wù)發(fā)現(xiàn)祭务,斷路器,智...
    卡卡羅2017閱讀 134,707評(píng)論 18 139
  • 坐在理發(fā)店的長凳上怪嫌,不知錯(cuò)過了幾批顧客义锥,直到老板娘大喊:“喂,你理發(fā)嗎岩灭?不理趕緊走拌倍!”我才抬起頭來,換個(gè)座位噪径,隨意...
    渺若恒沙閱讀 115評(píng)論 0 1
  • 今天跟大家分享一下我們?cè)谡{(diào)試中遇到的問題柱恤,個(gè)人覺得可能對(duì)我們處事方面會(huì)有所啟發(fā)。我們安裝的整瓶處理線瓶磚經(jīng)過...
    冠軍是誰_fcd3閱讀 258評(píng)論 0 0
  • 上面視頻的后面荚守,薛之謙即興唱的是: “是否說過要同去北方看雪景, 一晃如今已是十年前的約定练般, 我答應(yīng)過會(huì)在演唱會(huì)上...
    拾八日記閱讀 435評(píng)論 0 2