点击上方"IT牧场",选择"设为星标"

技术干货每日送达!

来源:http://t.cn/AiKuJEB9

  • 1. 引言

  • 2. AbstractRoutingDataSource

  • 3. 实践

    • 3.1. maven依赖

    • 3.2. 数据源配置

    • 3.3. 设置路由key / 查找数据源

  • 4. 测试

  • 5. 工程结构

1. 引言

读写分离要做的事情就是对于一条SQL该选择哪个数据库去执行,至于谁来做选择数据库这件事儿,无非两个,要么中间件帮我们做,要么程序自己做。因此,一般来讲,读写分离有两种实现方式。第一种是依靠中间件(比如:MyCat),也就是说应用程序连接到中间件,中间件帮我们做SQL分离;第二种是应用程序自己去做分离。这里我们选择程序自己来做,主要是利用Spring提供的路由数据源,以及AOP。

然而,应用程序层面去做读写分离最大的弱点(不足之处)在于无法动态增加数据库节点,因为数据源配置都是写在配置中的,新增数据库意味着新加一个数据源,必然改配置,并重启应用。当然,好处就是相对简单。

img

2. AbstractRoutingDataSource

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

img

3. 实践

关于配置请参考:

https://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.0modelVersion>

    <groupId>com.cjs.examplegroupId>    <artifactId>cjs-datasource-demoartifactId>    <version>0.0.1-SNAPSHOTversion>    <packaging>jarpackaging>

    <name>cjs-datasource-demoname>    <description>description>

    <parent>        <groupId>org.springframework.bootgroupId>        <artifactId>spring-boot-starter-parentartifactId>        <version>2.0.5.RELEASEversion>        <relativePath/>     parent>

    <properties>        <project.build.sourceEncoding>UTF-8project.build.sourceEncoding>        <project.reporting.outputEncoding>UTF-8project.reporting.outputEncoding>        <java.version>1.8java.version>    properties>

    <dependencies>        <dependency>            <groupId>org.springframework.bootgroupId>            <artifactId>spring-boot-starter-aopartifactId>        dependency>        <dependency>            <groupId>org.springframework.bootgroupId>            <artifactId>spring-boot-starter-jdbcartifactId>        dependency>        <dependency>            <groupId>org.springframework.bootgroupId>            <artifactId>spring-boot-starter-webartifactId>        dependency>        <dependency>            <groupId>org.mybatis.spring.bootgroupId>            <artifactId>mybatis-spring-boot-starterartifactId>            <version>1.3.2version>        dependency>        <dependency>            <groupId>org.apache.commonsgroupId>            <artifactId>commons-lang3artifactId>            <version>3.8version>        dependency>

        <dependency>            <groupId>mysqlgroupId>            <artifactId>mysql-connector-javaartifactId>            <scope>runtimescope>        dependency>        <dependency>            <groupId>org.springframework.bootgroupId>            <artifactId>spring-boot-starter-testartifactId>            <scope>testscope>        dependency>    dependencies>

    <build>        <plugins>            <plugin>                <groupId>org.springframework.bootgroupId>                <artifactId>spring-boot-maven-pluginartifactId>            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

多数据源配置

package com.cjs.example.config;

import com.cjs.example.bean.MyRoutingDataSource;import com.cjs.example.enums.DBTypeEnum;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.boot.context.properties.ConfigurationProperties;import org.springframework.boot.jdbc.DataSourceBuilder;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;

import javax.sql.DataSource;import java.util.HashMap;import java.util.Map;

/** * 关于数据源配置,参考SpringBoot官方文档第79章《Data Access》 * 79. Data Access * 79.1 Configure a Custom DataSource * 79.2 Configure Two DataSources */

@Configurationpublic 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 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配置

package com.cjs.example.config;

import org.apache.ibatis.session.SqlSessionFactory;import org.mybatis.spring.SqlSessionFactoryBean;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import org.springframework.core.io.support.PathMatchingResourcePatternResolver;import org.springframework.jdbc.datasource.DataSourceTransactionManager;import org.springframework.transaction.PlatformTransactionManager;import org.springframework.transaction.annotation.EnableTransactionManagement;

import javax.annotation.Resource;import javax.sql.DataSource;

@EnableTransactionManagement@Configurationpublic 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将数据源设置到每个线程上下文中

package com.cjs.example.bean;

import com.cjs.example.enums.DBTypeEnum;

import java.util.concurrent.atomic.AtomicInteger;

public class DBContextHolder {

    private static final ThreadLocal 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)

package com.cjs.example.aop;

import com.cjs.example.bean.DBContextHolder;import org.apache.commons.lang3.StringUtils;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 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

package com.cjs.example.service.impl;

import com.cjs.example.annotation.Master;import com.cjs.example.entity.Member;import com.cjs.example.entity.MemberExample;import com.cjs.example.mapper.MemberMapper;import com.cjs.example.service.MemberService;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import org.springframework.transaction.annotation.Transactional;

import java.util.List;

@Servicepublic 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 selectAll() {        return memberMapper.selectByExample(new MemberExample());    }

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

4. 测试

package com.cjs.example;

import com.cjs.example.entity.Member;import com.cjs.example.service.MemberService;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.boot.test.context.SpringBootTest;import org.springframework.test.context.junit4.SpringRunner;

@RunWith(SpringRunner.class)@SpringBootTestpublic 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");    }

}

查看控制台

img

干货分享

最近将个人学习笔记整理成册,使用PDF分享。关注我,回复如下代码,即可获得百度盘地址,无套路领取!

•001:《Java并发与高并发解决方案》学习笔记;•002:《深入JVM内核——原理、诊断与优化》学习笔记;•003:《Java面试宝典》•004:《Docker开源书》•005:《Kubernetes开源书》•006:《DDD速成(领域驱动设计速成)》•007:全部•008:加技术讨论群

往期精彩

•架构师眼中的高并发架构•手把手搭建生产可用的Nacos集群•抖音微博等短视频千万级高可用、高并发架构如何设计?•理解HTTP幂等性•微服务划分的姿势•MySQL 数据库优化,看这篇就够了


想知道更多?长按/扫码关注我吧↓↓↓>>>技术讨论群<<<喜欢就点个"在看"呗^_^

mysql 5.7和8.0区别_SpringBoot 2.0 教程实战 MySQL 读写分离相关推荐

  1. hibernate mysql 读写分离_SpringBoot集成Spring Data JPA及读写分离

    JPA是什么 JPA(Java Persistence API)是Sun官方提出的Java持久化规范,它为Java开发人员提供了一种对象/关联映射工具 来管理Java应用中的关系数据.它包括以下几方面 ...

  2. MySQL8.0 物理克隆接口_技术实战 MySQL 8.0.17 克隆插件分享-爱可生

    原标题:技术实战 MySQL 8.0.17 克隆插件分享-爱可生 背景 很神奇,5.7.17 和 8.0.17,连续两个17小版本都让人眼前一亮.前者加入了组复制(Group Replication) ...

  3. SpringBoot 2.0 教程实战 MySQL 读写分离

    点击上方 好好学java ,选择 星标 公众号 重磅资讯.干货,第一时间送达 今日推荐:2020,搞个 Mac 玩玩!个人原创+1博客:点击前往,查看更多 来源:https://www.cnblogs ...

  4. springboot jpa sql打印_SpringBoot集成Spring Data JPA以及读写分离

    相关代码:github OSCchina JPA是什么 JPA(Java Persistence API)是Sun官方提出的Java持久化规范,它为Java开发人员提供了一种对象/关联映射工具 来管理 ...

  5. 主从复制MySQL的安装和用数据库中间件MyCat实现分库分表、读写分离

    1.MySql主从复制 1.1.安装mysql 1.1.1.下载 下载地址:https://dev.mysql.com/downloads/mysql/ 1.1.2.卸载预装mysql #查看已安装: ...

  6. 定时任务-Quartz、Mycat简单入门、Linux下安装MySQL、Linux下安装MyCAT、Mycat的数据库分片、Mycat读写分离

    表现层:页面 后台管理系统.商城门户.搜索系统.订单系统.商品详情系统.购物车系统 中间件:dubbo 系统之间的通信,服务的统计,rpc协议远程过程调用 同步通信 服务层:实现具体的业务逻辑 商品服 ...

  7. MySQL集群系列2:通过keepalived实现双主集群读写分离

    在上一节基础上,通过添加keepalived实现读写分离. 首先关闭防火墙 安装keepalived keepalived 2台机器都要安装 rpm -q openssl kernel-devel y ...

  8. 【Java从0到架构师】项目实战 - 前后端分离、后端校验、Swagger、全局异常处理

    项目实战 - 前后端分离.后端校验.Swagger Layui 同源策略 SpringMVC 实现 CORS 后端校验 - hibernate-validator 方法的 Model 参数校验 方法的 ...

  9. mysql库与oracle库的区别_开源数据库Oracle与MySQL的SQL语法区别

    Oracle数据库与MySQL数据库的区别是本文我们主要要介绍的内容,接下来我们就开始介绍这部分内容,希望能够对您有所帮助. Oracle与MySQL的SQL语法区别: 1.在Oracle中用sele ...

最新文章

  1. 直方图_20210420
  2. 除了 iOS 和 Android,世界第三大移动系统是什么?
  3. Cocos坐标之convertToNodeSpace、convertToWorldSpace、convertToNodeSpaceAR、convertToWorldSpaceAR区别和用法...
  4. IB客座主编(一)--安普布线亚太区业务总监黎启枝
  5. 9.0 C++远征:对象成员
  6. typescript断言
  7. 操作系统:分享Win11几个实用小技巧,赶快收藏吧!
  8. 技术分享|单元测试推广与实战-在全新的DDD架构上进行单元测试
  9. project日历设置-大小周交替
  10. a as as big rat_超好玩!12句英语绕口令,你能一口气读完几句?
  11. linux ubuntu apache php 网站 'page not found'
  12. python微信语音转发方法_涨知识,微信语音能转发给别人啊,方法还那么简单
  13. 【车辆分类】基于matlab的视频中车辆跟踪监测分类算法仿真,包括背景差分与帧间差分以及形态学处理
  14. 计算机和电脑键盘进水怎么办,笔记本键盘进水了怎么办?处理笔记本电脑键盘进水的小妙招...
  15. 安装loadrunner时出现”命令行选项语法错误键入命令 \?获得帮助“的解决方法
  16. 利用html5画出五角星画出星空
  17. 给孩子简单快乐的童年
  18. 放弃幻想,人不会有什么长久安逸的
  19. steam游戏上架流程一:使用官方SDK上传游戏
  20. 计算机应用基础华工平时作业,计算机应用基础华工平时作业答案

热门文章

  1. 一台计算机连接两个投影,用一台PC控制四台投影 投影机多屏幕演示功能详解
  2. 计算实际例子_【科普】机器学习的核心计算:距离+统计?
  3. python面试技巧_经典7大Python面试题!看完考官竟然给了我30k的薪资
  4. 201632位matlab下载_【科研利器】带你get“研”途上的MATLAB入门篇
  5. phpstudy thinkphp5 mysql5.5+存储emoji
  6. 懒加载、瀑布流和LightBox实现图片搜索效果
  7. 清理清理火狐历史记录
  8. C++ string字符串的增删改查
  9. Linux下磁盘加密
  10. 每日英语:Chinese Show Global Real-Estate Appetite