一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - spring mybatis多數據源實例詳解

spring mybatis多數據源實例詳解

2020-05-29 14:31lqh JAVA教程

本文主要介紹sping mybatis多數據源處理,在開發過程中經常會遇到多個數據庫,這里給大家舉例說明如何處理,希望能幫助有需要的小伙伴

同一個項目有時會涉及到多個數據庫,也就是多數據源。多數據源又可以分為兩種情況:

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 多數據源處理的資料整理,希望能幫助有需要的朋友

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 午夜伦理:伦理片 | 美女扒开粉嫩尿口漫画 | 四虎精品成人免费视频 | 69日本xxxxxxxxx98 69人成网站色www | 97啪啪| 午夜AV国产欧美亚洲高清在线 | 国产综合色在线视频区色吧图片 | 4s4s4s4s色大众影视 | 俄罗斯激情性孕妇孕交大全 | 人人人人人看碰人人免费 | 无套日出白浆在线播放 | www亚洲色图| 日韩精品一区二区三区毛片 | 久久99精品久久久久久园产越南 | 97午夜| 日韩高清无砖砖区2022 | 亚洲人尿尿| 91一个人的在线观看www | 美女厕所尿尿擦逼 | 国产香蕉国产精品偷在线观看 | 天天做天天爱天天一爽一毛片 | 国产九九 | 亚洲一区二区三区不卡在线播放 | 欧美国产日韩在线播放 | 91麻豆网址 | 欧美日韩一区二区三区免费不卡 | 手机在线免费观看日本推理片 | leslessexvideos日本 | 欧美色fx性乌克兰 | www国产精品 | 国产一区二区三区四 | 女同xx美女放 | 亚洲 欧美 国产 日韩 字幕 | 久久国产热视频99rev6 | 日本韩国无矿砖码 | 亚洲高清中文字幕一区二区三区 | 青青青国产视频 | 国产精品视频久久久久 | 亚洲精品在线网址 | 亚洲一区二区三区在线播放 | 麻豆网站在线免费观看 |