簡(jiǎn)單三步手?jǐn)]springboot多數(shù)據(jù)源

在我們?nèi)粘i_發(fā)中大多數(shù)情況基本都是使用一個(gè)數(shù)據(jù)庫(kù)就完事了辑莫,但是也難免會(huì)遇到多數(shù)據(jù)庫(kù)的情況褐荷,這個(gè)時(shí)候多數(shù)據(jù)源就派上用場(chǎng)了弄痹。當(dāng)然了饭入,多數(shù)據(jù)源有很多的ORM框架都提供了方案,比如MybatisPlus肛真,MybatisFlex等谐丢,但是作為開發(fā)者的我們不僅要學(xué)會(huì)用輪子,也要知道一二吧蚓让,不能老拿著別人的輪子跑乾忱,不去了解和學(xué)習(xí)其原理吧。那么接下來(lái)历极,我就基于springboot分享一下如何定義自己的多數(shù)據(jù)源窄瘟。

第一步: 搭建springboot項(xiàng)目,配置yml, 比如我這里就定義兩個(gè)數(shù)據(jù)源即:db1與db2,注意這里和springboot默認(rèn)數(shù)據(jù)源配置有區(qū)別哦趟卸,是jdbc-url不是url蹄葱。

spring:
  datasource:
    db1:
      driver-class-name: com.mysql.cj.jdbc.Driver
      jdbc-url: jdbc:mysql://localhost:3306/db1?useUnicode=true&characterEncoding=utf-8
      username: 你的用戶名
      password: 你的密碼
    db2:
      driver-class-name: com.mysql.cj.jdbc.Driver
      jdbc-url: jdbc:mysql://localhost:3306/db2?useUnicode=true&characterEncoding=utf-8
      username: 你的用戶名
      password: 你的密碼

第二步: 開始擼碼
1.定義數(shù)據(jù)源枚舉,如還有可以繼續(xù)定義锄列。

public enum DataSourceType {
     //數(shù)據(jù)源1
    DB1,
    //數(shù)據(jù)源2
    DB2;
}

  1. 定義一個(gè)數(shù)據(jù)源切換上下文輔助類
 
public class DynamicDataSourceContextHolder {

    private static final ThreadLocal<String> LOCAL = new ThreadLocal<>();

   // 設(shè)置數(shù)據(jù)源
    public static void setDataSourceType(String type) {

       LOCAL.set(type);

    }
    //獲取數(shù)據(jù)源
    public static String getDataSourceType() {
        return LOCAL.get();
    }

   //清除數(shù)據(jù)源
    public static void clearDataSourceType() {
        LOCAL.remove();
    }
}

3.定義一個(gè)名為DynamicDatasource的類繼承spring提供的AbstractRoutingDataSource類

public class DynamicDataSource extends AbstractRoutingDataSource {

    //通過(guò)數(shù)據(jù)源上下文輔助類拿到數(shù)據(jù)源類型作為理由key
    @Override
    protected Object determineCurrentLookupKey() {
        return DynamicDataSourceContextHolder.getDataSourceType();
    }
}
  1. 注入數(shù)據(jù)源,排除springboot自動(dòng)配置數(shù)據(jù)源图云,不然啟動(dòng)會(huì)報(bào)找不到數(shù)據(jù)源的錯(cuò)誤
@Configuration
@EnableAutoConfiguration(exclude = {DataSourceAutoConfiguration.class})
public class DynamicConfig {

    // 數(shù)據(jù)1
    @Bean
    @ConfigurationProperties(prefix = "spring.datasource.db1")
    public DataSource db1DataSource(){

     return DataSourceBuilder.create().build();
    }
    // 數(shù)據(jù)2
    @Bean
    @ConfigurationProperties(prefix = "spring.datasource.db2")
    public DataSource db2DataSource(){

        return DataSourceBuilder.create().build();
    }

    //主數(shù)據(jù)源及數(shù)據(jù)源列表配置
    @Bean
    @Primary
    public DataSource dynamicDataSource(){

        final DynamicDataSource dynamicDataSource = new DynamicDataSource();

        final HashMap<Object, Object> targetDataSources = new HashMap<>(2);

        final DataSource db1DataSource = db1DataSource();
        final DataSource dbed2DataSource = db2DataSource();

        targetDataSources.put(DataSourceType.DB1.name(),db1DataSource);

        targetDataSources.put(DataSourceType.DB2.name(),dbed2DataSource);
        dynamicDataSource.setTargetDataSources(targetDataSources);

        //設(shè)置默認(rèn)數(shù)據(jù)源
        dynamicDataSource.setDefaultTargetDataSource(db1DataSource);

        return dynamicDataSource;
    }

}
  1. 至此就完成了代碼的編寫,接下來(lái)測(cè)試多數(shù)據(jù)源
 final List<User> list = userMapper.list();

        log.info("=====db1===={}",list);

  //通過(guò)上下文輔助類切換數(shù)據(jù)源     
 DynamicDataSourceContextHolder.setDataSourceType(DataSourceType.DB2.name());
        final List<User> list2 = userMapper.list();

        log.info("======db2==={}",list2);

運(yùn)行起能夠查詢兩個(gè)數(shù)據(jù)庫(kù)的數(shù)據(jù)就說(shuō)明ok了,功能上雖然沒(méi)有問(wèn)題了邻邮,但是還是不夠優(yōu)雅竣况,我們接下來(lái)要實(shí)現(xiàn)更優(yōu)雅的數(shù)據(jù)源切換。

第三步:快碼加鞭筒严,使用自定義注解加Aop實(shí)現(xiàn)更優(yōu)雅的數(shù)據(jù)源切換丹泉。

1.定義注解

@Target({ElementType.METHOD,ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface DataSource {
    //默認(rèn)數(shù)據(jù)源為db1
    DataSourceType value() default DataSourceType.DB1;
}

2.定義aop

@Aspect
@Component
public class DataSourceAop {


    /**
     * 定義切點(diǎn)為自定義注解
     */
    @Pointcut("@annotation(com.example.datasourcedemo.dynamic.annotation.DataSource)")
    public void pointCut(){

    }

    /**
     * 切換數(shù)據(jù)源
     * @param point
     */
    @Before("pointCut()")
    public void changeDataSource(JoinPoint point){

        final MethodSignature signature = (MethodSignature) point.getSignature();

        final Method method = signature.getMethod();

        final DataSource annotation = method.getAnnotation(DataSource.class);

        final DataSourceType type = annotation.value();
        
        if(!type.name().equals(DynamicDataSourceContextHolder.getDataSourceType())){
            DynamicDataSourceContextHolder.setDataSourceType(type.name());
        }

    }

    /**
     * 執(zhí)行后恢復(fù)為默認(rèn)數(shù)據(jù)源
     * @param point
     */
    @After("pointCut()")
    public void restDataSource(JoinPoint point){

        if(!DataSourceType.DB1.name().equals(DynamicDataSourceContextHolder.getDataSourceType())){
            DynamicDataSourceContextHolder.setDataSourceType(DataSourceType.DB1.name());
        }

    }
  1. 在測(cè)試的mapper上加上注解, 再進(jìn)行測(cè)試
@Mapper
public interface UserMapper {

    @Select("select * from user")
    List<User> list();
     //用上自定義注解
    @DataSource(DataSourceType.DB2)
    @Select("select * from user")
    List<User> list2();
}

去除之前測(cè)試代碼中的上下文輔助切換方式,再次運(yùn)行情萤,結(jié)果一致就說(shuō)明ok了。


 final List<User> list = userMapper.list();

        log.info("=====db1===={}",list);
 final List<User> list2 = userMapper.list2();

        log.info("=====db2===={}",list2);

總結(jié)摹恨,通過(guò)以上三步就完成了springboot自定義多數(shù)據(jù)源紫岩,是不是也沒(méi)有想象中那么難。輪子雖好睬塌,但是我們作為開發(fā)者也要學(xué)習(xí)實(shí)現(xiàn),不能老是拿來(lái)主義歇万,知其然不知其所以然揩晴,這樣永遠(yuǎn)也突破不了自己。今天就分享到這里了贪磺,有什么問(wèn)題歡迎留言探討或批評(píng)指正硫兰,如果喜歡我的文章記得關(guān)注我哦??!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末寒锚,一起剝皮案震驚了整個(gè)濱河市劫映,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌刹前,老刑警劉巖泳赋,帶你破解...
    沈念sama閱讀 206,968評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異喇喉,居然都是意外死亡祖今,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門拣技,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)千诬,“玉大人,你說(shuō)我怎么就攤上這事膏斤⌒彀螅” “怎么了?”我有些...
    開封第一講書人閱讀 153,220評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵莫辨,是天一觀的道長(zhǎng)傲茄。 經(jīng)常有香客問(wèn)我,道長(zhǎng)沮榜,這世上最難降的妖魔是什么烫幕? 我笑而不...
    開封第一講書人閱讀 55,416評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮敞映,結(jié)果婚禮上较曼,老公的妹妹穿的比我還像新娘。我一直安慰自己振愿,他們只是感情好捷犹,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,425評(píng)論 5 374
  • 文/花漫 我一把揭開白布弛饭。 她就那樣靜靜地躺著,像睡著了一般萍歉。 火紅的嫁衣襯著肌膚如雪侣颂。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評(píng)論 1 285
  • 那天枪孩,我揣著相機(jī)與錄音憔晒,去河邊找鬼。 笑死蔑舞,一個(gè)胖子當(dāng)著我的面吹牛拒担,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播攻询,決...
    沈念sama閱讀 38,432評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼从撼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了钧栖?” 一聲冷哼從身側(cè)響起低零,我...
    開封第一講書人閱讀 37,088評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎拯杠,沒(méi)想到半個(gè)月后掏婶,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,586評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡潭陪,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,028評(píng)論 2 325
  • 正文 我和宋清朗相戀三年气堕,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片畔咧。...
    茶點(diǎn)故事閱讀 38,137評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡茎芭,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出誓沸,到底是詐尸還是另有隱情梅桩,我是刑警寧澤,帶...
    沈念sama閱讀 33,783評(píng)論 4 324
  • 正文 年R本政府宣布拜隧,位于F島的核電站宿百,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏洪添。R本人自食惡果不足惜垦页,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,343評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望干奢。 院中可真熱鬧痊焊,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至垄惧,卻和暖如春刁愿,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背到逊。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工铣口, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人觉壶。 一個(gè)月前我還...
    沈念sama閱讀 45,595評(píng)論 2 355
  • 正文 我出身青樓脑题,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親掰曾。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,901評(píng)論 2 345

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