同一個項目有時會涉及到多個數據庫,也就是多數據源。多數據源又可以分為兩種情況:
1)兩個或多個數據庫沒有相關性,各自獨立,其實這種可以作為兩個項目來開發。比如在游戲開發中一個數據庫是平臺數據庫,其它還有平臺下的游戲對應的數據庫;
2)兩個或多個數據庫是master-slave的關系,比如有mysql搭建一個 master-master,其后又帶有多個slave;或者采用MHA搭建的master-slave復制;
目前我所知道的 Spring 多數據源的搭建大概有兩種方式,可以根據多數據源的情況進行選擇。
1. 采用spring配置文件直接配置多個數據源
比如針對兩個數據庫沒有相關性的情況,可以采用直接在spring的配置文件中配置多個數據源,然后分別進行事務的配置,如下所示:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
|
<context:component-scan base- package = "net.aazj.service,net.aazj.aop" /> <context:component-scan base- package = "net.aazj.aop" /> <!-- 引入屬性文件 --> <context:property-placeholder location= "classpath:config/db.properties" /> <!-- 配置數據源 --> <bean name= "dataSource" class = "com.alibaba.druid.pool.DruidDataSource" init-method= "init" destroy-method= "close" > <property name= "url" value= "${jdbc_url}" /> <property name= "username" value= "${jdbc_username}" /> <property name= "password" value= "${jdbc_password}" /> <!-- 初始化連接大小 --> <property name= "initialSize" value= "0" /> <!-- 連接池最大使用連接數量 --> <property name= "maxActive" value= "20" /> <!-- 連接池最大空閑 --> <property name= "maxIdle" value= "20" /> <!-- 連接池最小空閑 --> <property name= "minIdle" value= "0" /> <!-- 獲取連接最大等待時間 --> <property name= "maxWait" value= "60000" /> </bean> <bean id= "sqlSessionFactory" class = "org.mybatis.spring.SqlSessionFactoryBean" > <property name= "dataSource" ref= "dataSource" /> <property name= "configLocation" value= "classpath:config/mybatis-config.xml" /> <property name= "mapperLocations" value= "classpath*:config/mappers/**/*.xml" /> </bean> <!-- Transaction manager for a single JDBC DataSource --> <bean id= "transactionManager" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <property name= "dataSource" ref= "dataSource" /> </bean> <!-- 使用annotation定義事務 --> <tx:annotation-driven transaction-manager= "transactionManager" /> <bean class = "org.mybatis.spring.mapper.MapperScannerConfigurer" > <property name= "basePackage" value= "net.aazj.mapper" /> <property name= "sqlSessionFactoryBeanName" value= "sqlSessionFactory" /> </bean> <!-- Enables the use of the @AspectJ style of Spring AOP --> <aop:aspectj-autoproxy/> |
第二個數據源的配置
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
|
<bean name= "dataSource_2" class = "com.alibaba.druid.pool.DruidDataSource" init-method= "init" destroy-method= "close" > <property name= "url" value= "${jdbc_url_2}" /> <property name= "username" value= "${jdbc_username_2}" /> <property name= "password" value= "${jdbc_password_2}" /> <!-- 初始化連接大小 --> <property name= "initialSize" value= "0" /> <!-- 連接池最大使用連接數量 --> <property name= "maxActive" value= "20" /> <!-- 連接池最大空閑 --> <property name= "maxIdle" value= "20" /> <!-- 連接池最小空閑 --> <property name= "minIdle" value= "0" /> <!-- 獲取連接最大等待時間 --> <property name= "maxWait" value= "60000" /> </bean> <bean id= "sqlSessionFactory_slave" class = "org.mybatis.spring.SqlSessionFactoryBean" > <property name= "dataSource" ref= "dataSource_2" /> <property name= "configLocation" value= "classpath:config/mybatis-config-2.xml" /> <property name= "mapperLocations" value= "classpath*:config/mappers2/**/*.xml" /> </bean> <!-- Transaction manager for a single JDBC DataSource --> <bean id= "transactionManager_2" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <property name= "dataSource" ref= "dataSource_2" /> </bean> <!-- 使用annotation定義事務 --> <tx:annotation-driven transaction-manager= "transactionManager_2" /> <bean class = "org.mybatis.spring.mapper.MapperScannerConfigurer" > <property name= "basePackage" value= "net.aazj.mapper2" /> <property name= "sqlSessionFactoryBeanName" value= "sqlSessionFactory_2" /> </bean> |
如上所示,我們分別配置了兩個 dataSource,兩個sqlSessionFactory,兩個transactionManager,以及關鍵的地方在于 MapperScannerConfigurer 的配置——使用sqlSessionFactoryBeanName屬性,注入不同的sqlSessionFactory的名稱,這樣的話,就為不同的數 據庫對應的 mapper 接口注入了對應的 sqlSessionFactory。
需要注意的是,多個數據庫的這種配置是不支持分布式事務的,也就是同一個事務中,不能操作多個數據庫。這種配置方式的優點是很簡單,但是卻不靈 活。對于master-slave類型的多數據源配置而言不太適應,master-slave性的多數據源的配置,需要特別靈活,需要根據業務的類型進行 細致的配置。比如對于一些耗時特別大的select語句,我們希望放到slave上執行,而對于update,delete等操作肯定是只能在 master上執行的,另外對于一些實時性要求很高的select語句,我們也可能需要放到master上執行——比如一個場景是我去商城購買一件兵器, 購買操作的很定是master,同時購買完成之后,需要重新查詢出我所擁有的兵器和金幣,那么這個查詢可能也需要防止master上執行,而不能放在 slave上去執行,因為slave上可能存在延時,我們可不希望玩家發現購買成功之后,在背包中卻找不到兵器的情況出現。
所以對于master-slave類型的多數據源的配置,需要根據業務來進行靈活的配置,哪些select可以放到slave上,哪些select不能放到slave上。所以上面的那種所數據源的配置就不太適應了。
2. 基于 AbstractRoutingDataSource 和 AOP 的多數據源的配置
基本原理是,我們自己定義一個DataSource類ThreadLocalRountingDataSource,來繼承 AbstractRoutingDataSource,然后在配置文件中向ThreadLocalRountingDataSource注入 master 和 slave 的數據源,然后通過 AOP 來靈活配置,在哪些地方選擇 master 數據源,在哪些地方需要選擇 slave數據源。下面看代碼實現:
1)先定義一個enum來表示不同的數據源:
1
2
3
4
5
6
7
8
|
package net.aazj.enums; /** * 數據源的類別:master/slave */ public enum DataSources { MASTER, SLAVE } |
2)通過 TheadLocal 來保存每個線程選擇哪個數據源的標志(key):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
package net.aazj.util; import net.aazj.enums.DataSources; public class DataSourceTypeManager { private static final ThreadLocal<DataSources> dataSourceTypes = new ThreadLocal<DataSources>(){ @Override protected DataSources initialValue(){ return DataSources.MASTER; } }; public static DataSources get(){ return dataSourceTypes.get(); } public static void set(DataSources dataSourceType){ dataSourceTypes.set(dataSourceType); } public static void reset(){ dataSourceTypes.set(DataSources.MASTER0); } } |
3)定義 ThreadLocalRountingDataSource,繼承AbstractRoutingDataSource:
1
2
3
4
5
6
7
8
9
10
|
package net.aazj.util; import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; public class ThreadLocalRountingDataSource extends AbstractRoutingDataSource { @Override protected Object determineCurrentLookupKey() { return DataSourceTypeManager.get(); } } |
4)在配置文件中向 ThreadLocalRountingDataSource 注入 master 和 slave 的數據源:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
|
<context:component-scan base- package = "net.aazj.service,net.aazj.aop" /> <context:component-scan base- package = "net.aazj.aop" /> <!-- 引入屬性文件 --> <context:property-placeholder location= "classpath:config/db.properties" /> <!-- 配置數據源Master --> <bean name= "dataSourceMaster" class = "com.alibaba.druid.pool.DruidDataSource" init-method= "init" destroy-method= "close" > <property name= "url" value= "${jdbc_url}" /> <property name= "username" value= "${jdbc_username}" /> <property name= "password" value= "${jdbc_password}" /> <!-- 初始化連接大小 --> <property name= "initialSize" value= "0" /> <!-- 連接池最大使用連接數量 --> <property name= "maxActive" value= "20" /> <!-- 連接池最大空閑 --> <property name= "maxIdle" value= "20" /> <!-- 連接池最小空閑 --> <property name= "minIdle" value= "0" /> <!-- 獲取連接最大等待時間 --> <property name= "maxWait" value= "60000" /> </bean> <!-- 配置數據源Slave --> <bean name= "dataSourceSlave" class = "com.alibaba.druid.pool.DruidDataSource" init-method= "init" destroy-method= "close" > <property name= "url" value= "${jdbc_url_slave}" /> <property name= "username" value= "${jdbc_username_slave}" /> <property name= "password" value= "${jdbc_password_slave}" /> <!-- 初始化連接大小 --> <property name= "initialSize" value= "0" /> <!-- 連接池最大使用連接數量 --> <property name= "maxActive" value= "20" /> <!-- 連接池最大空閑 --> <property name= "maxIdle" value= "20" /> <!-- 連接池最小空閑 --> <property name= "minIdle" value= "0" /> <!-- 獲取連接最大等待時間 --> <property name= "maxWait" value= "60000" /> </bean> <bean id= "dataSource" class = "net.aazj.util.ThreadLocalRountingDataSource" > <property name= "defaultTargetDataSource" ref= "dataSourceMaster" /> <property name= "targetDataSources" > <map key-type= "net.aazj.enums.DataSources" > <entry key= "MASTER" value-ref= "dataSourceMaster" /> <entry key= "SLAVE" value-ref= "dataSourceSlave" /> <!-- 這里還可以加多個dataSource --> </map> </property> </bean> <bean id= "sqlSessionFactory" class = "org.mybatis.spring.SqlSessionFactoryBean" > <property name= "dataSource" ref= "dataSource" /> <property name= "configLocation" value= "classpath:config/mybatis-config.xml" /> <property name= "mapperLocations" value= "classpath*:config/mappers/**/*.xml" /> </bean> <!-- Transaction manager for a single JDBC DataSource --> <bean id= "transactionManager" class = "org.springframework.jdbc.datasource.DataSourceTransactionManager" > <property name= "dataSource" ref= "dataSource" /> </bean> <!-- 使用annotation定義事務 --> <tx:annotation-driven transaction-manager= "transactionManager" /> <bean class = "org.mybatis.spring.mapper.MapperScannerConfigurer" > <property name= "basePackage" value= "net.aazj.mapper" /> <!-- <property name= "sqlSessionFactoryBeanName" value= "sqlSessionFactory" /> --> </bean> |
上面spring的配置文件中,我們針對master數據庫和slave數據庫分別定義了dataSourceMaster和 dataSourceSlave兩個dataSource,然后注入到<bean id="dataSource" class="net.aazj.util.ThreadLocalRountingDataSource"> 中,這樣我們的dataSource就可以來根據 key 的不同來選擇dataSourceMaster和 dataSourceSlave了。
5)使用Spring AOP 來指定 dataSource 的 key ,從而dataSource會根據key選擇 dataSourceMaster 和 dataSourceSlave:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
package net.aazj.aop; import net.aazj.enums.DataSources; import net.aazj.util.DataSourceTypeManager; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; @Aspect // for aop @Component // for auto scan public class DataSourceInterceptor { @Pointcut ( "execution(public * net.aazj.service..*.getUser(..))" ) public void dataSourceSlave(){}; @Before ( "dataSourceSlave()" ) public void before(JoinPoint jp) { DataSourceTypeManager.set(DataSources.SLAVE); } // ... ... } |
這里我們定義了一個 Aspect 類,我們使用 @Before 來在符合 @Pointcut("execution(public * net.aazj.service..*.getUser(..))") 中的方法被調用之前,調用 DataSourceTypeManager.set(DataSources.SLAVE) 設置了 key 的類型為 DataSources.SLAVE,所以 dataSource 會根據key=DataSources.SLAVE 選擇 dataSourceSlave 這個dataSource。所以該方法對于的sql語句會在slave數據庫上執行。
我們可以不斷的擴充 DataSourceInterceptor 這個 Aspect,在中進行各種各樣的定義,來為某個service的某個方法指定合適的數據源對應的dataSource。
這樣我們就可以使用 Spring AOP 的強大功能來,十分靈活進行配置了。
6)AbstractRoutingDataSource原理剖析
ThreadLocalRountingDataSource 繼承了 AbstractRoutingDataSource, 實現其抽象方法 protected abstract Object determineCurrentLookupKey(); 從而實現對不同數據源的路由功能。我們從源碼入手分析下其中原理:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
public abstract class AbstractRoutingDataSource extends AbstractDataSource implements InitializingBean AbstractRoutingDataSource 實現了 InitializingBean 那么spring在初始化該bean時,會調用InitializingBean的接口 void afterPropertiesSet() throws Exception; 我們看下AbstractRoutingDataSource是如何實現這個接口的: @Override public void afterPropertiesSet() { if ( this .targetDataSources == null ) { throw new IllegalArgumentException( "Property 'targetDataSources' is required" ); } this .resolvedDataSources = new HashMap<Object, DataSource>( this .targetDataSources.size()); for (Map.Entry<Object, Object> entry : this .targetDataSources.entrySet()) { Object lookupKey = resolveSpecifiedLookupKey(entry.getKey()); DataSource dataSource = resolveSpecifiedDataSource(entry.getValue()); this .resolvedDataSources.put(lookupKey, dataSource); } if ( this .defaultTargetDataSource != null ) { this .resolvedDefaultDataSource = resolveSpecifiedDataSource( this .defaultTargetDataSource); } } |
targetDataSources 是我們在xml配置文件中注入的 dataSourceMaster 和 dataSourceSlave. afterPropertiesSet方法就是使用注入的。
dataSourceMaster 和 dataSourceSlave來構造一個HashMap——resolvedDataSources。方便后面根據 key 從該map 中取得對應的dataSource。
我們在看下 AbstractDataSource 接口中的 Connection getConnection() throws SQLException; 是如何實現的:
1
2
3
4
|
@Override public Connection getConnection() throws SQLException { return determineTargetDataSource().getConnection(); } |
關鍵在于 determineTargetDataSource(),根據方法名就可以看出,應該此處就決定了使用哪個 dataSource :
1
2
3
4
5
6
7
8
9
10
11
12
|
protected DataSource determineTargetDataSource() { Assert.notNull( this .resolvedDataSources, "DataSource router not initialized" ); Object lookupKey = determineCurrentLookupKey(); 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 + "]" ); } return dataSource; } |
Object lookupKey = determineCurrentLookupKey(); 該方法是我們實現的,在其中獲取ThreadLocal中保存的 key 值。獲得了key之后,在從afterPropertiesSet()中初始化好了的resolvedDataSources這個map中獲得key對應的dataSource。而ThreadLocal中保存的 key 值 是通過AOP的方式在調用service中相關方法之前設置好的。OK,到此搞定!
3. 總結
從本文中我們可以體會到AOP的強大和靈活。
以上就是sping,mybatis 多數據源處理的資料整理,希望能幫助有需要的朋友