一平超凡 | 作者
urlify.cn/jEnuIn | 来源
1、背景
在实际的项目中,一般一个项目都会有主数据库和从数据库,主从数据库之间的数据同步是通过数据库的配置来完成的,一般地这个工作都是由DBA来进行完成。但是,如果我们的项目中的业务量比较大的时候,我们希望读操作从数据库中读取数据,写操作的时候才将数据保存至主数据库,然后主数据库和从数据库之间通过通信将数据完成同步;那么,我们的程序是如何将做到读操作的时候从从库中读取数据,写操作的时候是如何将数据写入到主库的呢?这个问题,就是今天要解决的问题;
目前市面上实现主从数据源切换的方式主要有两种,一种是利用第三方插件的形式实现,另外一种就是通过使用AOP进行实现。我采用的实现方式就是利用SpringAOP的方式实现;
2、实现
2.1 导入所需要的依赖包
我的项目使用的SpringBoot实现的,ORM框架使用的是Mybatis,数据源使用的是阿里的Druid。配置如下:
- 2.2 配置数据源
为了节约成本,我只在本地的计算机上进行了代码实现,所以我只是在本地的同一个mysql服务上配置了多个数据库,数据库之间也没有进行主从的配置,毕竟我的主要目的是想看看代码的实现效果;配置文件如下:
# 主库spring.datasource.master.name=masterspring.datasource.master.driver-class-name=com.mysql.jdbc.Driverspring.datasource.master.url=jdbc:mysql://localhost:3306/master?serverTimezone=UTC&useSSL=falsespring.datasource.master.username=rootspring.datasource.master.password=root# 从库1spring.datasource.slaver1.name=slaver1spring.datasource.slaver1.driver-class-name=com.mysql.jdbc.Driverspring.datasource.slaver1.url=jdbc:mysql://localhost:3306/slaver1?serverTimezone=UTC&useSSL=falsespring.datasource.slaver1.username=rootspring.datasource.slaver1.password=root# 从库2spring.datasource.slaver2.name=slaver1spring.datasource.slaver2.driver-class-name=com.mysql.jdbc.Driverspring.datasource.slaver2.url=jdbc:mysql://localhost:3306/slaver2?serverTimezone=UTC&useSSL=falsespring.datasource.slaver2.username=rootspring.datasource.slaver2.password=root
2.3 业务操作层的实现
数据的操作需要借助Service层和Dao层的进行实现,由于这部分不是实现主从数据源的关键部分,所以此处的代码就不进行展示;
2.4 数据源配置
我们都知道,Spring和Mybatis在整合的时候都需要配置 org.mybatis.spring.SqlSessionFactoryBean 的实例,在配置这个实例的时候需要指定数据源。那么如果想要实现主从数据源动态切换的功能,这个数据源的配置就不能使用传统的DataSource了,这里我是用的是 org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource 数据源。这个数据源是Spring提供的,它可以在获取数据源连接之前通过方法 determineTargetDataSource() 判断获取哪一个数据源的连接;也正是因为这个特性,我们才得以实现数据源动态切换的功能;
数据源配置如下:
import com.alibaba.druid.spring.boot.autoconfigure.DruidDataSourceBuilder;import com.example.demo.config.db.DataSourceRoutingDataSource;import org.mybatis.spring.SqlSessionFactoryBean;import org.springframework.boot.context.properties.ConfigurationProperties;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.context.annotation.Primary;import org.springframework.core.io.Resource;import org.springframework.core.io.support.PathMatchingResourcePatternResolver;import org.springframework.stereotype.Component;import javax.sql.DataSource;import java.io.IOException;import java.util.HashMap;import java.util.HashSet;import java.util.Map;import java.util.Set;/** * 数据源配置文件 */@Component@Configurationpublic class DatasourceConfig { /** * 创建一个主数据源的实例 */ @Primary @Bean(value = "master") @ConfigurationProperties(prefix = "spring.datasource.master") public DataSource master() { return DruidDataSourceBuilder.create().build(); } /** * 从数据源1 */ @Bean(value = "slaver1") @ConfigurationProperties(prefix = "spring.datasource.slaver1") public DataSource slaver1() { return DruidDataSourceBuilder.create().build(); } /** * 从数据源2 */ @Bean(value = "slaver2") @ConfigurationProperties(prefix = "spring.datasource.slaver2") public DataSource slaver2() { return DruidDataSourceBuilder.create().build(); } /** * DataSourceRoutingDataSource 继承了 AbstractRoutingDataSource; * 主要为了实现determineCurrentLookupKey()方法; */ @Bean(value = "dataSource") public DataSourceRoutingDataSource dataSource() { DataSourceRoutingDataSource dataSource = new DataSourceRoutingDataSource(); // 数据源 Map dataSources = new HashMap<>(); dataSources.put("master", master()); dataSources.put("slaver1", slaver1()); dataSources.put("slaver2", slaver2()); dataSource.setTargetDataSources(dataSources); dataSource.setDefaultTargetDataSource(master()); // 设置主数据源的键值; Set masterKeys = new HashSet<>(); masterKeys.add("master"); dataSource.setMasterKeys(masterKeys); // 设置从数据源的键值; Set slaverKeys = new HashSet<>(); slaverKeys.add("slaver1"); slaverKeys.add("slaver2"); dataSource.setSlaverKeys(slaverKeys); return dataSource; } /** * SqlSessionFactoryBean实例配置 */ @Bean public SqlSessionFactoryBean sqlSessionFactoryBean() throws IOException { SqlSessionFactoryBean factoryBean = new SqlSessionFactoryBean(); factoryBean.setDataSource(dataSource()); PathMatchingResourcePatternResolver resolver = new PathMatchingResourcePatternResolver(); Resource[] resources = resolver.getResources("classpath*:mapper/*.xml"); factoryBean.setMapperLocations(resources); factoryBean.setTypeAliasesPackage("com.example.demo.domain"); return factoryBean; }}
DataSourceRoutingDataSource实现如下:
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;import java.util.ArrayList;import java.util.List;import java.util.Set;import java.util.concurrent.atomic.AtomicBoolean;import java.util.concurrent.atomic.AtomicInteger;public class DataSourceRoutingDataSource extends AbstractRoutingDataSource { public static AtomicBoolean MASTER_STATUS = new AtomicBoolean(true); private static List MASTER_KEYS = new ArrayList<>(); private static AtomicInteger MASTER_INDEX = new AtomicInteger(0); private static List SLAVER_KEYS = new ArrayList<>(); private static AtomicInteger SLAVER_INDEX = new AtomicInteger(0); /* * 关键点:用于切换数据源 * */ @Override protected Object determineCurrentLookupKey() { if (MASTER_STATUS.get()) { return getNextMaster(); } else { return getNextSlaver(); } } public void setMasterKeys(Set masterKeys) { MASTER_KEYS.addAll(masterKeys); } public void setSlaverKeys(Set slaverKeys) { SLAVER_KEYS.addAll(slaverKeys); } /** * 获取下一个主库的key */ private Object getNextMaster() { if (MASTER_KEYS.size() == 1) { return MASTER_KEYS.get(0); } int index = MASTER_INDEX.getAndAdd(1); return MASTER_KEYS.get(index % MASTER_KEYS.size()); } /** * 获取下一个从库的key */ private Object getNextSlaver() { if (SLAVER_KEYS.size() == 1) { return SLAVER_KEYS.get(0); } int index = SLAVER_INDEX.getAndAdd(1); return SLAVER_KEYS.get(index % SLAVER_KEYS.size()); }}
2.5 AOP配置
实现上面的步骤其实已经可以进行增删改查的功能了,但是我们目的不在此;我们还要通过AOP进行数据源的切换,所以我们还需要配置AOP;我这里写的比较简单,就是根据service的名称判断是否使用主库;代码如下:
import com.example.demo.config.db.DataSourceRoutingDataSource;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@Componentpublic class ServiceAspect { @Pointcut(value = "execution(* com.example.demo.service.*.*(..))") public void point() {} @Before(value = "point()") public void before(JoinPoint joinPoint) { String name = joinPoint.getSignature().getName(); if (name.startsWith("get") || name.startsWith("query") || name.startsWith("find")) { DataSourceRoutingDataSource.MASTER_STATUS.set(false); } else { DataSourceRoutingDataSource.MASTER_STATUS.set(true); } }}
import com.example.demo.config.db.DataSourceRoutingDataSource;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@Componentpublic class ServiceAspect { @Pointcut(value = "execution(* com.example.demo.service.*.*(..))") public void point() {} @Before(value = "point()") public void before(JoinPoint joinPoint) { String name = joinPoint.getSignature().getName(); if (name.startsWith("get") || name.startsWith("query") || name.startsWith("find")) { DataSourceRoutingDataSource.MASTER_STATUS.set(false); } else { DataSourceRoutingDataSource.MASTER_STATUS.set(true); } }}
3、总结
动态数据源的实现方式还是比较简单的,核心就在于配置数据源为 org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource 类型的数据源;如果感兴趣的话,可以看一看内部的实现源码;