返回顶部
首页 > 资讯 > 数据库 >如何编写简单的demo实现读写分离
  • 122
分享到

如何编写简单的demo实现读写分离

2024-04-02 19:04:59 122人浏览 独家记忆
摘要

本篇内容主要讲解“如何编写简单的demo实现读写分离”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何编写简单的demo实现读写分离”吧! 前言相信有

本篇内容主要讲解“如何编写简单的demo实现读写分离”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何编写简单的demo实现读写分离”吧!

如何编写简单的demo实现读写分离

 前言

相信有经验的同学都清楚,当db的读写量过高时,我们会备份一份或多份的从库用于做数据的读取,然后主库就主要承担写入的功能(也有读取需要,但压力不大),当db分好主从库后,我们还需要在项目实现自动连接主从库,达到读写分离的效果。实现读写分离并不困难,只要在数据库连接池手动控制好对应的db服务地址即可,但那样就会侵入业务代码,而且一个项目操作数据库的地方可能很多,如果都手动控制的话无疑会是很大的工作量,对此,我们有必要改造出一套方便的工具

以Java语言来说,如今大部分的项目都是基于spring  Boot框架来搭建项目架构的,结合Spring本身自带的aop工具,我们可以很容易就构建能实现读写分离效果的注解类,用注解的话可以达到对业务代码无入侵的效果,而且使用上也比较方便。

下面就简单带大家写个demo。

环境部署

数据库:Mysql

库数量:2个,一主一从

关于mysql的主从环境部署网上有很多文章可以参考,这里不做介绍了。

开始项目

首先,毫无疑问,先开始搭建一个SpringBoot工程,然后在pom文件中引入如下依赖:

<dependencies>         <dependency>             <groupId>com.alibaba</groupId>             <artifactId>druid-spring-boot-starter</artifactId>             <version>1.1.10</version>         </dependency>         <dependency>             <groupId>org.mybatis.spring.boot</groupId>             <artifactId>mybatis-spring-boot-starter</artifactId>             <version>1.3.2</version>         </dependency>         <dependency>             <groupId>tk.mybatis</groupId>             <artifactId>mapper-spring-boot-starter</artifactId>             <version>2.1.5</version>         </dependency>         <dependency>             <groupId>mysql</groupId>             <artifactId>mysql-connector-java</artifactId>             <version>8.0.16</version>         </dependency>         <!-- 动态数据源 所需依赖 ### start-->         <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter-jdbc</artifactId>             <scope>provided</scope>         </dependency>         <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter-aop</artifactId>             <scope>provided</scope>         </dependency>         <!-- 动态数据源 所需依赖 ### end-->         <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter-WEB</artifactId>         </dependency>      <dependency>             <groupId>org.projectlombok</groupId>             <artifactId>lombok</artifactId>             <optional>true</optional>         </dependency>         <dependency>             <groupId>com.alibaba</groupId>             <artifactId>fastJSON</artifactId>             <version>1.2.4</version>         </dependency>         <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter-test</artifactId>             <scope>test</scope>         </dependency>         <dependency>             <groupId>org.springframework.boot</groupId>             <artifactId>spring-boot-starter-data-jpa</artifactId>         </dependency>     </dependencies>

目录结构

引入基本的依赖后,整理一下目录结构,完成后的项目骨架大致如下:

如何编写简单的demo实现读写分离

建表

创建一张表user,在主库执行sql语句同时在从库生成对应的表数据

DROP TABLE IF EXISTS `user`; CREATE TABLE `user` (   `user_id` bigint(20) NOT NULL COMMENT '用户id',   `user_name` varchar(255) DEFAULT '' COMMENT '用户名称',   `user_phone` varchar(50) DEFAULT '' COMMENT '用户手机',   `address` varchar(255) DEFAULT '' COMMENT '住址',   `weight` int(3) NOT NULL DEFAULT '1' COMMENT '权重,大者优先',   `created_at` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',   `updated_at` datetime DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',   PRIMARY KEY (`user_id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8;  INSERT INTO `user` VALUES ('1196978513958141952', '测试1', '18826334748', '广州市海珠区', '1', '2019-11-20 10:28:51', '2019-11-22 14:28:26'); INSERT INTO `user` VALUES ('1196978513958141953', '测试2', '18826274230', '广州市天河区', '2', '2019-11-20 10:29:37', '2019-11-22 14:28:14'); INSERT INTO `user` VALUES ('1196978513958141954', '测试3', '18826273900', '广州市天河区', '1', '2019-11-20 10:30:19', '2019-11-22 14:28:30');

主从数据源配置

application.yml,主要信息是主从库的数据源配置

server:   port: 8001 spring:   jackson:    date-fORMat: yyyy-MM-dd HH:mm:ss    time-zone: GMT+8   datasource:     type: com.alibaba.druid.pool.DruidDataSource     driver-class-name: com.mysql.cj.jdbc.Driver     master:       url: jdbc:mysql://127.0.0.1:3307/user?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&failOverReadOnly=false&useSSL=false&zeroDateTimeBehavior=convertToNull&allowMultiQueries=true       username: root       passWord:     slave:       url: jdbc:mysql://127.0.0.1:3308/user?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=UTF-8&autoReconnect=true&failOverReadOnly=false&useSSL=false&zeroDateTimeBehavior=convertToNull&allowMultiQueries=true       username: root       password:

因为有一主一从两个数据源,我们用枚举类来代替,方便我们使用时能对应

@Getter public enum DynamicDataSourceEnum {     MASTER("master"),     SLAVE("slave");     private String dataSourceName;     DynamicDataSourceEnum(String dataSourceName) {         this.dataSourceName = dataSourceName;     } }

数据源配置信息类 DataSourceConfig,这里配置了两个数据源,masterDb和slaveDb

@Configuration @MapperScan(basePackages = "com.xjt.proxy.mapper", sqlSessionTemplateRef = "sqlTemplate") public class DataSourceConfig {           // 主库       @Bean       @ConfigurationProperties(prefix = "spring.datasource.master")       public DataSource masterDb() {   return DruidDataSourceBuilder.create().build();       }           @Bean     @ConditionalOnProperty(prefix = "spring.datasource", name = "slave", matchIfMissing = true)     @ConfigurationProperties(prefix = "spring.datasource.slave")     public DataSource slaveDb() {         return DruidDataSourceBuilder.create().build();     }           @Bean     public DynamicDataSource dynamicDb(@Qualifier("masterDb") DataSource masterDataSource,         @Autowired(required = false) @Qualifier("slaveDb") DataSource slaveDataSource) {         DynamicDataSource dynamicDataSource = new DynamicDataSource();         Map<Object, Object> targetDataSources = new HashMap<>();         targetDataSources.put(DynamicDataSourceEnum.MASTER.getDataSourceName(), masterDataSource);         if (slaveDataSource != null) {             targetDataSources.put(DynamicDataSourceEnum.SLAVE.getDataSourceName(), slaveDataSource);         }         dynamicDataSource.setTargetDataSources(targetDataSources);         dynamicDataSource.setDefaultTargetDataSource(masterDataSource);         return dynamicDataSource;     }     @Bean     public SqlSessionFactory sessionFactory(@Qualifier("dynamicDb") DataSource dynamicDataSource) throws Exception {         SqlSessionFactoryBean bean = new SqlSessionFactoryBean();         bean.setMapperLocations(             new PathMatchingResourcePatternResolver().getResources("classpath*:mapper/*Mapper.xml"));         bean.setDataSource(dynamicDataSource);         return bean.getObject();     }     @Bean     public SqlSessionTemplate sqlTemplate(@Qualifier("sessionFactory") SqlSessionFactory sqlSessionFactory) {         return new SqlSessionTemplate(sqlSessionFactory);     }     @Bean(name = "dataSourceTx")     public DataSourceTransactionManager dataSourceTx(@Qualifier("dynamicDb") DataSource dynamicDataSource) {         DataSourceTransactionManager dataSourceTransactionManager = new DataSourceTransactionManager();         dataSourceTransactionManager.setDataSource(dynamicDataSource);         return dataSourceTransactionManager;     } }

设置路由

设置路由的目的为了方便查找对应的数据源,我们可以用ThreadLocal保存数据源的信息到每个线程中,方便我们需要时获取

public class DataSourceContextHolder {     private static final ThreadLocal<String> DYNAMIC_DATASOURCE_CONTEXT = new ThreadLocal<>();     public static void set(String datasourceType) {         DYNAMIC_DATASOURCE_CONTEXT.set(datasourceType);     }     public static String get() {         return DYNAMIC_DATASOURCE_CONTEXT.get();     }     public static void clear() {         DYNAMIC_DATASOURCE_CONTEXT.remove();     } }

获取路由

public class DynamicDataSource extends AbstractRoutingDataSource {     @Override     protected Object determineCurrentLookupKey() {         return DataSourceContextHolder.get();     } }

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

数据源的注解

为了可以方便切换数据源,我们可以写一个注解,注解中包含数据源对应的枚举值,默认是主库,

@Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) @Documented public @interface DataSourceSelector {      DynamicDataSourceEnum value() default DynamicDataSourceEnum.MASTER;     boolean clear() default true; }

aop切换数据源

到这里,aop终于可以现身出场了,这里我们定义一个aop类,对有注解的方法做切换数据源的操作,具体代码如下:

@Slf4j @Aspect @Order(value = 1) @Component public class DataSourceContextAop {   @Around("@annotation(com.xjt.proxy.dynamicdatasource.DataSourceSelector)")     public Object setDynamicDataSource(ProceedingJoinPoint pjp) throws Throwable {         boolean clear = true;         try {             Method method = this.getMethod(pjp);             DataSourceSelector dataSourceImport = method.getAnnotation(DataSourceSelector.class);             clear = dataSourceImport.clear();             DataSourceContextHolder.set(dataSourceImport.value().getDataSourceName());             log.info("========数据源切换至:{}", dataSourceImport.value().getDataSourceName());             return pjp.proceed();         } finally {             if (clear) {                 DataSourceContextHolder.clear();             }          }     }     private Method getMethod(JoinPoint pjp) {         MethodSignature signature = (MethodSignature)pjp.getSignature();         return signature.getMethod();     }  }

到这一步,我们的准备配置工作就完成了,下面开始测试效果。

先写好Service文件,包含读取和更新两个方法,

@Service public class UserService {      @Autowired     private UserMapper userMapper;      @DataSourceSelector(value = DynamicDataSourceEnum.MASTER)     public int update(Long userId) {         User user = new User();         user.setUserId(userId);         user.setUserName("老薛");         return userMapper.updateByPrimaryKeySelective(user);     }      @DataSourceSelector(value = DynamicDataSourceEnum.SLAVE)     public User find(Long userId) {         User user = new User();         user.setUserId(userId);         return userMapper.selectByPrimaryKey(user);     } }

根据方法上的注解可以看出,读的方法走从库,更新的方法走主库,更新的对象是userId为1196978513958141952 的数据,

然后我们写个测试类测试下是否能达到效果,

@RunWith(SpringRunner.class) @SpringBootTest class UserServiceTest {      @Autowired     UserService userService;      @Test     void find() {         User user = userService.find(1196978513958141952L);         System.out.println("id:" + user.getUserId());         System.out.println("name:" + user.getUserName());         System.out.println("phone:" + user.getUserPhone());     }      @Test     void update() {         Long userId = 1196978513958141952L;         userService.update(userId);         User user = userService.find(userId);         System.out.println(user.getUserName());     }  }

测试结果:

1、读取方法

如何编写简单的demo实现读写分离

2、更新方法

如何编写简单的demo实现读写分离

执行之后,比对数据库就可以发现主从库都修改了数据,说明我们的读写分离是成功的。当然,更新方法可以指向从库,这样一来就只会修改到从库的数据,而不会涉及到主库。

到此,相信大家对“如何编写简单的demo实现读写分离”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

您可能感兴趣的文档:

--结束END--

本文标题: 如何编写简单的demo实现读写分离

本文链接: https://lsjlt.com/news/61601.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 如何编写简单的demo实现读写分离
    本篇内容主要讲解“如何编写简单的demo实现读写分离”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“如何编写简单的demo实现读写分离”吧! 前言相信有...
    99+
    2024-04-02
  • Mycat简单实现读写分离与分库分表
    Mycat数据库读写分离 环境: 客户端1.13 ↓ mycat中间件1.11 ↙ ...
    99+
    2024-04-02
  • maxscale + mariadb5.5如何实现读写分离
    本篇文章给大家分享的是有关maxscale + mariadb5.5如何实现读写分离,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。1、安装[r...
    99+
    2024-04-02
  • MySQL中如何实现读写分离
    MySQL中如何实现读写分离,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。一,创建Master数据库的配置文件vi master...
    99+
    2024-04-02
  • php+mysql如何实现读写分离
    这篇“php+mysql如何实现读写分离”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“php+mysql如何实现读写分离”文...
    99+
    2023-07-05
  • Atlas实现读写分离
    该Atlas方案的实现需要基于MHA架构(而MHA架构需要 实现mysql主从复制且开启GTID特性) 常见方案介绍: Mysql-proxy(oracle)Mysql-router(oracle)Atla...
    99+
    2024-04-02
  • ProxySQL实现读写分离
    环境: 192.168.205.37: as ProxySQL server 192.168.205.47: as Master server 192.168.205.57: as Slave ser...
    99+
    2024-04-02
  • mysql中Oneproxy如何实现读写分离
    这篇文章将为大家详细讲解有关mysql中Oneproxy如何实现读写分离,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。基本架构: 写请求全部定向到主库,数据通过日志异步复...
    99+
    2024-04-02
  • SpringBoot详解如何实现读写分离
    目录前言1.项目引入依赖2.yml配置3.启动4.测试5.中间所遇到的问题前言 根据公司业务需求,项目需要读写分离,所以记录下读写分离的过程。 分为两个部分: 1.项目的读写分离。 ...
    99+
    2024-04-02
  • Mysql数据库读写分离简单配置
    环境:Master:192.168.71.128      mysql-sql-node1Slave:192.168.71.140 &n...
    99+
    2024-04-02
  • MyCAT实现MySQL的读写分离
    在MySQL中间件出现之前,对于MySQL主从集群,如果要实现其读写分离,一般是在程序端实现,这样就带来一个问题,即数据库和程序的耦合度太高,如果我数据库的地址发生改变了,那么我程序端也要进行相应的修改,如...
    99+
    2024-04-02
  • mysql如何读写分离
    mysql 读写分离是一种数据库架构,将数据库分为主库和从库,主库负责写入操作,从库负责读操作,以降低主库负载和提高并发读能力。实现 mysql 读写分离需要:搭建主从复制环境、修改应用...
    99+
    2024-04-14
    mysql
  • redis如何读写分离
    redis 通过主从复制和客户端配置实现读写分离。优点包括提高读吞吐量、保证写入一致性和提高可用性。需要注意数据一致性、配置复杂性和适用于高读写负载场景。 Redis 如何实现读写分离...
    99+
    2024-04-19
    redis
  • mysql+mycat实现读写分离
    centos7 master slave mycat1.6 client 192.168.41.10 192.168.41.11 192.168.41.12 192.168.41.13 ...
    99+
    2024-04-02
  • mysql基于amoeba如何实现读写分离
    下文给大家带来关于mysql基于amoeba如何实现读写分离,感兴趣的话就一起来看看这篇文章吧,相信看完mysql基于amoeba如何实现读写分离对大家多少有点帮助吧。环境:    &n...
    99+
    2024-04-02
  • Mycat如何实现Mysql集群读写分离
    这篇文章给大家分享的是有关Mycat如何实现Mysql集群读写分离的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。MySQL 读写分离的概述MySQL 作为目前世界上使用最广泛的免...
    99+
    2024-04-02
  • SpringBoot详解MySQL如何实现读写分离
    目录前言一、主从数据源的配置二、数据源路由的配置三、数据源上下文环境四、切换注解和Aop配置五、用法以及测试六、总结前言 首先思考一个问题:在高并发的场景中,关于数据库都有哪些优化的...
    99+
    2024-04-02
  • SpringBoot使用JPA如何实现读写分离
    今天就跟大家聊聊有关SpringBoot使用JPA如何实现读写分离,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。JPA是什么JPA(Java Persistence API)是Sun...
    99+
    2023-05-31
    springboot jpa 读写分离
  • MySQL Router实现MySQL的读写分离
    MySQL Router实现MySQL的读写分离 https://www.cnblogs.com/f-ck-need-u/p/9276639.html https://www.cnblogs.com...
    99+
    2024-04-02
  • Redis主从实现读写分离
    前言 大家在工作中可能会遇到这样的需求,即Redis读写分离,目的是为了压力分散化。下面我将为大家介绍借助AWS的ELB实现读写分离,以写主读从为例。 实现 引用库文件 <!-- redis客...
    99+
    2022-06-04
    主从 Redis
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作