0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

读写分离的两种实现方式

Android编程精选 来源:博客园 作者:狂乱的贵公子 2021-09-29 14:51 次阅读

1. 引言

读写分离要做的事情就是对于一条SQL该选择哪个数据库去执行,至于谁来做选择数据库这件事儿,无非两个,要么中间件帮我们做,要么程序自己做。因此,一般来讲,读写分离有两种实现方式。

第一种是依靠中间件(比如:MyCat),也就是说应用程序连接到中间件,中间件帮我们做SQL分离;第二种是应用程序自己去做分离。这里我们选择程序自己来做,主要是利用Spring提供的路由数据源,以及AOP。然而,应用程序层面去做读写分离最大的弱点(不足之处)在于无法动态增加数据库节点,因为数据源配置都是写在配置中的,新增数据库意味着新加一个数据源,必然改配置,并重启应用。当然,好处就是相对简单。

2. AbstractRoutingDataSource

基于特定的查找key路由到特定的数据源。它内部维护了一组目标数据源,并且做了路由key与目标数据源之间的映射,提供基于key查找数据源的方法。

3. 实践

关于配置请参考《MySQL主从复制配置》

地址:www.cnblogs.com/cjsblog/p/9706370.html

3.1. maven依赖

《?xml version=“1.0” encoding=“UTF-8”?》 《project xmlns=“http://maven.apache.org/POM/4.0.0” xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance” xsi:schemaLocation=“http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd”》 《modelVersion》4.0.0《/modelVersion》

《groupId》com.cjs.example《/groupId》 《artifactId》cjs-datasource-demo《/artifactId》 《version》0.0.1-SNAPSHOT《/version》 《packaging》jar《/packaging》 《name》cjs-datasource-demo《/name》 《description》《/description》 《parent》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-starter-parent

《/artifactId》 《version》2.0.5.RELEASE《/version》 《relativePath/》 《!-- lookup parent from repository --》 《/parent》 《properties》 《project.build.sourceEncoding》UTF-8《/project.build.sourceEncoding》 《project.reporting.outputEncoding》

UTF-8

《/project.reporting.outputEncoding》 《java.version》1.8《/java.version》 《/properties》 《dependencies》 《dependency》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-starter-aop《/artifactId》 《/dependency》

《dependency》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-starter-jdbc《/artifactId》 《/dependency》 《dependency》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-starter-web《/artifactId》

《/dependency》 《dependency》 《groupId》org.mybatis.spring.boot《/groupId》 《artifactId》mybatis-spring-boot-starter《/artifactId》 《version》

1.3.2《/version》 《/dependency》 《dependency》 《groupId》org.apache.commons《/groupId》 《artifactId》commons-lang3《/artifactId》 《version》3.8《/version》 《/dependency》 《dependency》

《groupId》mysql《/groupId》 《artifactId》mysql-connector-java《/artifactId》 《scope》runtime《/scope》 《/dependency》 《dependency》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-starter-test《/artifactId》 《scope》test《/scope》 《/dependency》 《/dependencies》 《build》 《plugins》 《plugin》 《groupId》org.springframework.boot《/groupId》 《artifactId》spring-boot-maven-plugin《/artifactId》 《/plugin》 《

!--《plugin》 《groupId》org.mybatis.generator《/groupId》 《artifactId》mybatis-generator-maven-plugin《/artifactId》 《version》1.3.5《/version》 《dependencies》 《dependency》 《groupId》mysql《/groupId》 《artifactId》mysql-connector-java《/artifactId》 《version》

5.1.46《/version》 《/dependency》 《/dependencies》 《configuration》 《configurationFile》

${basedir}/src/main/resources/myBatisGeneratorConfig.xml《/configurationFile》 《overwrite》true《/overwrite》 《/configuration》 《executions》 《execution》 《id》Generate MyBatis Artifacts《/id》 《goals》 《goal》generate《/goal》 《/goals》 《/execution》 《/executions》 《/plugin》--》 《/plugins》 《/build》 《/project》

3.2. 数据源配置application.yml

spring: datasource: master: jdbc-url: jdbc:mysql://192.168.102.31:3306/test username: root password: 123456 driver-class-name: com.mysql.jdbc.Driver slave1: jdbc-url: jdbc:mysql://192.168.102.56:3306/test username:

pig # 只读账户 password: 123456 driver-class-name: com.mysql.jdbc.Driver slave2: jdbc-url: jdbc:mysql://192.168.102.36:3306/test username: pig # 只读账户 password: 123456 driver-class-name: com.mysql.jdbc.Driver多数据源配置

/** * 关于数据源配置,参考SpringBoot官方文档第79章《Data Access》

* 79. Data Access * 79.1 Configure a Custom DataSource * 79.2 Configure Two DataSources */ @Configuration public class DataSourceConfig

{ @Bean @ConfigurationProperties(“spring.datasource.master”) public DataSource masterDataSource()

{ return DataSourceBuilder.create().build(); } @Bean @ConfigurationProperties

(“spring.datasource.slave1”) public DataSource slave1DataSource() { return DataSourceBuilder.create().build(); } @Bean @ConfigurationProperties

(“spring.datasource.slave2”) public DataSource slave2DataSource() { return DataSourceBuilder.create().build(); } @Bean public DataSource myRoutingDataSource(@Qualifier

(“masterDataSource”) DataSource masterDataSource, @Qualifier(“slave1DataSource”) DataSource slave1DataSource, @Qualifier(“slave2DataSource”) DataSource slave2DataSource) { Map《Object, Object》 targetDataSources = new HashMap《》(); targetDataSources.put

(DBTypeEnum.MASTER, masterDataSource); targetDataSources.put(DBTypeEnum.SLAVE1, slave1DataSource); targetDataSources.put

(DBTypeEnum.SLAVE2, slave2DataSource); MyRoutingDataSource myRoutingDataSource = new MyRoutingDataSource(); myRoutingDataSource.setDefaultTargetDataSource(masterDataSource); myRoutingDataSource.setTargetDataSources(targetDataSources);

return myRoutingDataSource; } }这里,我们配置了4个数据源,1个master,2两个slave,1个路由数据源。前3个数据源都是为了生成第4个数据源,而且后续我们只用这最后一个路由数据源。MyBatis配置

@EnableTransactionManagement @Configuration public class MyBatisConfig { @Resource(name = “myRoutingDataSource”) private DataSource myRoutingDataSource;

@Bean public SqlSessionFactory sqlSessionFactory() throws Exception { SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();

sqlSessionFactoryBean.setDataSource(myRoutingDataSource); sqlSessionFactoryBean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources(“classpath:mapper/*.xml”));

return sqlSessionFactoryBean.getObject(); } @Bean public PlatformTransactionManager platformTransactionManager() { return new DataSourceTransactionManager(myRoutingDataSource);

} }由于 Spring 容器中现在有4个数据源,所以我们需要为事务管理器和MyBatis手动指定一个明确的数据源。3.3. 设置路由key / 查找数据源目标数据源就是那前3个这个我们是知道的,但是使用的时候是如果查找数据源的呢?首先,我们定义一个枚举来代表这三个数据源

package com.cjs.example.enums; public enum DBTypeEnum { MASTER, SLAVE1, SLAVE2; }接下来,通过ThreadLocal将数据源设置到每个线程上下文中

public class DBContextHolder { private static final ThreadLocal《DBTypeEnum》 contextHolder = new ThreadLocal《》();

private static final AtomicInteger counter = new AtomicInteger(-1); public static void set(DBTypeEnum dbType) { contextHolder.set(dbType); } public static DBTypeEnum get() { return contextHolder.get();

} public static void master() { set(DBTypeEnum.MASTER); System.out.println(“切换到master”);

} public static void slave() { // 轮询 int index = counter.getAndIncrement() % 2; if (counter.get() 》 9999) { counter.set(-1); } if (index == 0) { set(DBTypeEnum.SLAVE1); System.out.println(“切换到slave1”); }else { set(DBTypeEnum.SLAVE2);

System.out.println(“切换到slave2”); } } }获取路由key

package com.cjs.example.bean; import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource; import org.springframework.lang.Nullable; public class MyRoutingDataSource extends AbstractRoutingDataSource { @Nullable @Override protected Object determineCurrentLookupKey()

{ return DBContextHolder.get(); } }设置路由key默认情况下,所有的查询都走从库,插入/修改/删除走主库。我们通过方法名来区分操作类型(CRUD)

@Aspect @Component public class DataSourceAop { @Pointcut(“!@annotation(com.cjs.example.annotation.Master) ” + “&& (execution(* com.cjs.example.service..*.select*(。。)) ” + “|| execution(* com.cjs.example.service..*.get*(。。)))”)

public void readPointcut() { } @Pointcut(“@annotation(com.cjs.example.annotation.Master) ” + “|| execution(* com.cjs.example.service..*.insert*(。。)) ” + “|| execution(* com.cjs.example.service..*.add*(。。)) ” + “|| execution(* com.cjs.example.service..*.update*(。。))

” + “|| execution(* com.cjs.example.service..*.edit*(。。)) ” + “|| execution(* com.cjs.example.service..*.delete*(。。)) ” + “|| execution(* com.cjs.example.service..*.remove*(。。))”) public void writePointcut()

{ } @Before(“readPointcut()”) public void read() { DBContextHolder.slave(); } @Before(“writePointcut()”) public void write() { DBContextHolder.master(); } /** * 另一种写法:if.。.else.。。 判断哪些需要读从数据库,其余的走主数据库 */ // @Before

(“execution(* com.cjs.example.service.impl.*.*(。。))”) // public void before(JoinPoint jp) { // String methodName = jp.getSignature().getName();

// // if (StringUtils.startsWithAny(methodName, “get”, “select”, “find”)) { // DBContextHolder.slave();

// }else { // DBContextHolder.master(); // } // } }有一般情况就有特殊情况,特殊情况是某些情况下我们需要强制读主库,针对这种情况,我们定义一个主键,用该注解标注的就读主库

package com.cjs.example.annotation; public @interface Master { }例如,假设我们有一张表member

@Service public class MemberServiceImpl implements MemberService { @Autowired private MemberMapper memberMapper; @Transactional @Override public int insert(Member member) { return memberMapper.insert(member);

} @Master @Override public int save(Member member) { return memberMapper.insert(member);

} @Override public List《Member》 selectAll() { return memberMapper.selectByExample(new MemberExample()); } @Master @Override public String getToken(String appId) { // 有些读操作必须读主数据库 // 比如,获取微信access_token,因为高峰时期主从同步可能延迟 // 这种情况下就必须强制从主数据读 return null; } }4. 测试

@RunWith(SpringRunner.class) @SpringBootTest public class CjsDatasourceDemoApplicationTests { @Autowired private MemberService memberService; @Test public void testWrite() { Member member = new Member(); member.setName(“zhangsan”); memberService.insert(member);

} @Test public void testRead() { for (int i = 0; i 《 4; i++) { memberService.selectAll(); } } @Test public void testSave() { Member member = new Member(); member.setName(“wangwu”); memberService.save(member);

} @Test public void testReadFromMaster() { memberService.getToken(“1234”); } }查看控制台

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • MySQL
    +关注

    关注

    1

    文章

    773

    浏览量

    25976
  • Sprint
    +关注

    关注

    0

    文章

    86

    浏览量

    15032

原文标题:Spring Boot + MyBatis + MySQL读写分离

文章出处:【微信号:AndroidPush,微信公众号:Android编程精选】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    自定义系统时钟频率后,用库函数和微带两种方式点亮LED灯,为什么灯的闪烁频率不一致?

    写了一个自定义系统时钟频率的函数,然后自定义系统时钟频率为28MHz,在这个频率下用库函数和微带两种方式点亮LED灯,为什么灯的闪烁频率不一致? 代码在附件中,哪位大神帮忙看下是什么原因?*附件:点亮LED.zip
    发表于 03-25 21:45

    变频器的两种无速度传感器矢量控制模式有什么区别?

    在变频器控制模式中,有无速度矢量传感器模式,该模式又分为两种,无感矢量0和无感矢量1,这两种模式的本质原理是什么?对变频器的控制电机性能和系统稳定性响应什么的有什么影响?具体应用上哪种方式更好一点,或者而言这
    发表于 02-22 21:45

    异或门两种常见的实现方式

    两种实现方式都能够实现异或门的功能,具体的选择取决于设计需求和逻辑门的可用性。实际构建异或门时,可以使用离散电子元件(如晶体管、二极管等)或整合电路芯片(如 TTL、CMOS 等)来
    的头像 发表于 02-04 17:30 3607次阅读
    异或门<b class='flag-5'>两种</b>常见的<b class='flag-5'>实现</b><b class='flag-5'>方式</b>

    redis两种持久化方式的区别

    的完整性和一致性。 Redis提供了两种持久化方式:RDB(Redis Database)和AOF(Append Only File)。这两种方式各有优劣,下面我们将详细介绍它们的区别
    的头像 发表于 12-04 11:12 242次阅读

    Python如何使用MySQL 8.2读写分离

    如您所知,MySQL 8.2 发布了最令人期待的功能之一:读写分离
    的头像 发表于 11-22 09:39 268次阅读
    Python如何使用MySQL 8.2<b class='flag-5'>读写</b><b class='flag-5'>分离</b>?

    avr单片机两种烧写程序方式的区别?

    最近在做一个单片机程序,有点疑问,用avrstudio烧程序接的是avrdragon,用ISPUSB是另外一个软件,这两种烧程序有什么区别?者过程都是先写熔丝,写flash,写eeprom;我试着用ISPUSB连avrsrudio,没有选择成功,有点疑惑。
    发表于 11-10 07:41

    DC-DC和LDO两种供电模式有什么优缺点?

    给单片机供电的电路经常见到DC-DC 和 LDO 两种供电方式,这个选型上有什么规律吗,各有什么样的优缺点
    发表于 11-01 06:43

    常见的ARM架构分为两种是M系列另外一是A系列,这两种有什么区别啊?

    现在市面上常见的ARM架构分为两种是M系列另外一是A系列,这两种有什么区别啊,用的时候他们一般分别用在什么地方啊。
    发表于 10-26 07:00

    阐述DDR3读写分离的方法

    DDR3是2007年推出的,预计2022年DDR3的市场份额将降至8%或以下。但原理都是一样的,DDR3的读写分离作为DDR最基本也是最常用的部分,本文主要阐述DDR3读写分离的方法。
    的头像 发表于 10-18 16:03 565次阅读
    阐述DDR3<b class='flag-5'>读写</b><b class='flag-5'>分离</b>的方法

    请问stm32cube低功耗能不能两种唤醒同时用?

    stm32F103 设置成最低功耗的standy模式,使用RTC和PA0两种模式同时唤醒,实验RTC可以实现了,再加上PA0能不能双唤醒?
    发表于 09-25 07:00

    有没有办法将内置LED和GPIO2分离,以便我可以将它们用于两种不同的用途?

    LED 也会响应。似乎它们以某种方式耦合在一起。 有没有办法将内置 LED 和 GPIO2 分离,以便我可以将它们用于两种不同的用途?
    发表于 06-06 07:24

    星三角的两种控制方式

    今天给大家展示一下星三角的两种控制控制方式的电路图,PLC 控制方式以三菱PLC 为例
    发表于 05-05 15:53 439次阅读
    星三角的<b class='flag-5'>两种</b>控制<b class='flag-5'>方式</b>

    在PADS中怎么放置两种不同孔径的过孔啊?

    在PADS中怎么放置两种不同孔径的过孔啊?
    发表于 04-28 16:45

    互锁正反转电路的两种实现方式

    有关正反转电路的知识,互锁正反转电路的两种实现方式,一种是接触器互锁正反转,一种是按钮互锁正反转,学习下这两种电路的工作原理,下面一起来看下。
    的头像 发表于 04-27 15:47 6874次阅读
    互锁正反转电路的<b class='flag-5'>两种</b><b class='flag-5'>实现</b><b class='flag-5'>方式</b>