您好,登錄后才能下訂單哦!
SpringBoot中動(dòng)態(tài)切換數(shù)據(jù)源的方法是怎樣的,相信很多沒有經(jīng)驗(yàn)的人對(duì)此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。
其實(shí)這個(gè)表示有點(diǎn)不太對(duì),應(yīng)該是 Druid 動(dòng)態(tài)切換數(shù)據(jù)源的方法,只是應(yīng)用在了 springboot 框架中,準(zhǔn)備代碼準(zhǔn)備了半天,之前在一次數(shù)據(jù)庫(kù)遷移中使用了,發(fā)現(xiàn) Druid 還是很強(qiáng)大的,用來做動(dòng)態(tài)數(shù)據(jù)源切換很方便。
首先這里的場(chǎng)景跟我原來用的有點(diǎn)點(diǎn)區(qū)別,在項(xiàng)目中使用的是通過配置中心控制數(shù)據(jù)源切換,統(tǒng)一切換,而這里的例子多加了個(gè)可以根據(jù)接口注解配置
第一部分是最核心的,如何基于 Spring JDBC 和 Druid 來實(shí)現(xiàn)數(shù)據(jù)源切換,是繼承了org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource 這個(gè)類,他的determineCurrentLookupKey方法會(huì)被調(diào)用來獲得用來決定選擇那個(gè)數(shù)據(jù)源的對(duì)象,也就是 lookupKey,也可以通過這個(gè)類看到就是通過這個(gè) lookupKey 來路由找到數(shù)據(jù)源。
public class DynamicDataSource extends AbstractRoutingDataSource { @Override protected Object determineCurrentLookupKey() { if (DatabaseContextHolder.getDatabaseType() != null) { return DatabaseContextHolder.getDatabaseType().getName(); } return DatabaseType.MASTER1.getName(); } }
而如何使用這個(gè) lookupKey 呢,就涉及到我們的 DataSource 配置了,原來就是我們可以直接通過spring 的 jdbc 配置數(shù)據(jù)源,像這樣
現(xiàn)在我們要使用 Druid 作為數(shù)據(jù)源了,然后配置 DynamicDataSource的參數(shù),通過 key 來選擇對(duì)應(yīng)的 DataSource,也就是下面配的 master1 和 master2
<bean id="master1" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close" p:driverClassName="com.mysql.cj.jdbc.Driver" p:url="${master1.demo.datasource.url}" p:username="${master1.demo.datasource.username}" p:password="${master1.demo.datasource.password}" p:initialSize="5" p:minIdle="1" p:maxActive="10" p:maxWait="60000" p:timeBetweenEvictionRunsMillis="60000" p:minEvictableIdleTimeMillis="300000" p:validationQuery="SELECT 'x'" p:testWhileIdle="true" p:testOnBorrow="false" p:testOnReturn="false" p:poolPreparedStatements="false" p:maxPoolPreparedStatementPerConnectionSize="20" p:connectionProperties="config.decrypt=true" p:filters="stat,config"/> <bean id="master2" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close" p:driverClassName="com.mysql.cj.jdbc.Driver" p:url="${master2.demo.datasource.url}" p:username="${master2.demo.datasource.username}" p:password="${master2.demo.datasource.password}" p:initialSize="5" p:minIdle="1" p:maxActive="10" p:maxWait="60000" p:timeBetweenEvictionRunsMillis="60000" p:minEvictableIdleTimeMillis="300000" p:validationQuery="SELECT 'x'" p:testWhileIdle="true" p:testOnBorrow="false" p:testOnReturn="false" p:poolPreparedStatements="false" p:maxPoolPreparedStatementPerConnectionSize="20" p:connectionProperties="config.decrypt=true" p:filters="stat,config"/> <bean id="dataSource" class="com.nicksxs.springdemo.config.DynamicDataSource"> <property name="targetDataSources"> <map key-type="java.lang.String"> <!-- master --> <entry key="master1" value-ref="master1"/> <!-- slave --> <entry key="master2" value-ref="master2"/> </map> </property> <property name="defaultTargetDataSource" ref="master1"/> </bean>
現(xiàn)在就要回到頭上,介紹下這個(gè)DatabaseContextHolder,這里使用了 ThreadLocal 存放這個(gè) DatabaseType,為啥要用這個(gè)是因?yàn)榍懊嬲f的我們想要讓接口層面去配置不同的數(shù)據(jù)源,要把持相互隔離不受影響,就使用了 ThreadLocal,關(guān)于它也可以看我前面寫的一篇文章聊聊傳說中的 ThreadLocal,而 DatabaseType 就是個(gè)簡(jiǎn)單的枚舉
public class DatabaseContextHolder { public static final ThreadLocal<DatabaseType> databaseTypeThreadLocal = new ThreadLocal<>(); public static DatabaseType getDatabaseType() { return databaseTypeThreadLocal.get(); } public static void putDatabaseType(DatabaseType databaseType) { databaseTypeThreadLocal.set(databaseType); } public static void clearDatabaseType() { databaseTypeThreadLocal.remove(); } } public enum DatabaseType { MASTER1("master1", "1"), MASTER2("master2", "2"); private final String name; private final String value; DatabaseType(String name, String value) { this.name = name; this.value = value; } public String getName() { return name; } public String getValue() { return value; } public static DatabaseType getDatabaseType(String name) { if (MASTER2.name.equals(name)) { return MASTER2; } return MASTER1; } }
這邊可以看到就是通過動(dòng)態(tài)地通過putDatabaseType設(shè)置lookupKey來進(jìn)行數(shù)據(jù)源切換,要通過接口注解配置來進(jìn)行設(shè)置的話,我們就需要一個(gè)注解
@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface DataSource { String value(); }
這個(gè)注解可以配置在我的接口方法上,比如這樣
public interface StudentService { @DataSource("master1") public Student queryOne(); @DataSource("master2") public Student queryAnother(); }
通過切面來進(jìn)行數(shù)據(jù)源的設(shè)置
@Aspect @Component @Order(-1) public class DataSourceAspect { @Pointcut("execution(* com.nicksxs.springdemo.service..*.*(..))") public void pointCut() { } @Before("pointCut()") public void before(JoinPoint point) { Object target = point.getTarget(); System.out.println(target.toString()); String method = point.getSignature().getName(); System.out.println(method); Class<?>[] classz = target.getClass().getInterfaces(); Class<?>[] parameterTypes = ((MethodSignature) point.getSignature()) .getMethod().getParameterTypes(); try { Method m = classz[0].getMethod(method, parameterTypes); System.out.println("method"+ m.getName()); if (m.isAnnotationPresent(DataSource.class)) { DataSource data = m.getAnnotation(DataSource.class); System.out.println("dataSource:"+data.value()); DatabaseContextHolder.putDatabaseType(DatabaseType.getDatabaseType(data.value())); } } catch (Exception e) { e.printStackTrace(); } } @After("pointCut()") public void after() { DatabaseContextHolder.clearDatabaseType(); } }
通過接口判斷是否帶有注解跟是注解的值,DatabaseType 的配置不太好,不過先忽略了,然后在切點(diǎn)后進(jìn)行清理
這是我 master1 的數(shù)據(jù),
master2 的數(shù)據(jù)
然后跑一下簡(jiǎn)單的 demo,
@Override public void run(String...args) { LOGGER.info("run here"); System.out.println(studentService.queryOne()); System.out.println(studentService.queryAnother()); }
看一下運(yùn)行結(jié)果
其實(shí)這個(gè)方法應(yīng)用場(chǎng)景不止可以用來遷移數(shù)據(jù)庫(kù),還能實(shí)現(xiàn)精細(xì)化的讀寫數(shù)據(jù)源分離之類的,算是做個(gè)簡(jiǎn)單記錄和分享。
看完上述內(nèi)容,你們掌握SpringBoot中動(dòng)態(tài)切換數(shù)據(jù)源的方法是怎樣的的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。