搭建基于springboot轻量级读写分离开发框架
何为读写分离
读写分离是指对资源的修改和读取进行分离,能解决很多数据库瓶颈,以及代码混乱难以维护等相关的问题,使系统有更好的扩展性,维护性和可用性。
一般会分三个步骤来实现:
一. 主从数据库搭建
信息管理系统的绝大部分瓶颈在数据库,通过搭建主从数据库,写到主数据库,读取从数据库,提高数据库的吞吐量,根据业务需求可以搭建一主一从、一主多从的数据库同步架构。如果报表多的系统,可以搭个一主多从架构,一个从数据库供普通查询,另一个从数据库供报表查询,这样能够避免报表的复杂查询影响客户正常操作。
二. 读写代码分离
代码上对读写进行分离。读的逻辑相对简单,几乎不需要做过多的分层封装。大部分业务逻辑在写操作,所以我们需要专注于对写代码的分层、抽象封装。注意: 在写模块涉及到业务数据读取,几乎要实时的,而且基于高内聚的原则,应该封装进写代码类中,读取主数据库。
三. 进程分离
将读和写的代码封装到不同的进程,从进程级别避免相互影响,其实就是分布式。实现从进程上解耦,程序运行期间的性能、异常错误不会相互影响,所以系统有相对高的可用性。
这里多说一句, 如果对写业务按领域拆分到不同的进程,会涉及到分布式事务,在未涉及到高并发、大数据的系统,其实没必要从进程上拆分,分布式对事务不友好,为了处理分布式事务,你需要付出更多的时间和金钱成本。考虑进程拆分,一定要基于实际业务需求再三权衡利弊。很多时候,也许你只需要多一个从数据库、一个缓存、多一台服务器、多几G内存、多几核cpu、优化一下sql 即可解决很多性能上的问题。
如何搭建
现在我们搭建一主一从数据库架构, 并且实现从代码上进行读写分离的开发框架,但不涉及进程分离。
1. 搭建主从数据库
mariadb 可参考搭建 mariadb 数据库主从同步 或者 https://mariadb.com/kb/en/setting-up-replication/ 。
2. 基于springboot 搭建开发框架
2.1 项目结构
画一下框架的模块结构

- api 模块相当于 gateway, 接收和响应请求,还包括鉴权, 参数的校验和组装,调用 command, query 的接口。
- common 模块封装一些和业务无关的通用功能类。
- query 是读模块,封装非实时的查询接口,查询"从数据库"。
- command 是写模块,封装领域的业务逻辑,操作"主数据库"。
按照上图,用 idea 创建项目结构如下

对 command 和 query 模块再进行细化

因为 command 模块我们使用领域驱动开发,所以拆分成服务(Service), 仓储(Repository), ORM, 聚合根(Aggregate)。
Query 只是简单的查询,我们直接用 Dao 访问数据库,然后把数据转成 DTO 返回。
根据上图,再细化项目的文档结构

2.2 配置文件
假设我们有三个环境, 分别是开发(dev), 测试(uat), 生产(prod)。每个模块都有单独的配置文件。
api:
application-api-dev.yml
application-api-uat.yml
application-api-prod.yml
command:
application-command-dev.yml
application-command-uat.yml
application-command-prod.yml
query:
application-query-dev.yml
application-query-uat.yml
application-query-prod.yml
在系统启动时,指定使用的环境, api 作为启动项目,添加 bootstrap.yml, 因为 bootstrap.yml 优先于 application.yml 生效,所以可以在 bootstrap.yml 配置启动环境。
我们启用 dev 环境, bootstrap.yml 内容如下:
spring:
profiles:
active: common-dev,command-dev,query-dev,api-dev
那么,对应的 application-common-dev.yml, application-command-dev.yml, application-query-dev.yml, application-api-dev.yml 配置文件将起效。
2.3 运行
在 query 项目添加一个接口
public interface UserQueryService {
String getName(Long id);
}
并实现它
@Service
public class UserQueryServiceImpl implements UserQueryService {
@Override
public String getName(Long id) {
return "my name is grissom" + id;
}
}
在 api 中调用该接口
@RestController
@RequestMapping("user")
public class UserController {
private final UserQueryService userQueryService;
public UserController(UserQueryService userQueryService) {
this.userQueryService = userQueryService;
}
@GetMapping("/name/{id}")
public String name(@PathVariable("id") Long id) {
return this.userQueryService.getName(id);
}
}
将 api 作为启动项,配置 application-api-dev.yml, 开放 8003 端口
server:
port: 8003
用 postman 请求

至此,咱们的项目结构已经搭建好。
下一篇再写如何访问数据库。
源码
https://github.com/grissomlau/cqrs-springboot
搭建基于springboot轻量级读写分离开发框架的更多相关文章
- 搭建基于MySQL的读写分离工具Amoeba
搭建基于MySQL的读写分离工具Amoeba: Amoeba工具是实现MySQL数据库读写分离的一个工具,前提是基于MySQL主从复制来实现的: 实验环境(虚拟机): 主机 角色 10.10.10.2 ...
- Mycat搭建负载均衡,读写分离的Mysql集群
Mycat搭建负载均衡,读写分离的Mysql集群 准备环境 1.mysql-5.7.24-linux-glibc2.12-x86_64.tar.gz 2.Mycat-server-1.6.7.4-te ...
- 基于SpringBoot前后端分离的点餐系统
基于SpringBoot前后端分离的点餐系统 开发环境:主要采用Spring boot框架和小程序开发 项目简介:点餐系统,分成卖家端和买家端.买家端使用微信小程序开发,实现扫码点餐.浏览菜单.下单. ...
- SpringBoot数据库读写分离之基于Docker构建主从数据库同步实例
看了好久的SpringBoot结合MyBatista实现读写,但是一直没有勇气实现他,今天终于接触到了读写分离的东西,读写分离就是讲读操作执行在Slave数据库(从数据库),写操作在Master数据库 ...
- Mysql8.0主从复制搭建,shardingsphere+springboot+mybatis读写分离
1.安装mysql8.0 首先需要在192.167.3.171上安装JDK. 下载mysql安装包,https://dev.mysql.com/downloads/,找到以下页面下载. 下载后放到li ...
- springboot实现读写分离(基于Mybatis,mysql)
近日工作任务较轻,有空学习学习技术,遂来研究如果实现读写分离.这里用博客记录下过程,一方面可备日后查看,同时也能分享给大家(网上的资料真的大都是抄来抄去,,还不带格式的,看的真心难受). 完整代码:h ...
- SpringBoot Mybatis 读写分离配置(山东数漫江湖)
为什么需要读写分离 当项目越来越大和并发越来大的情况下,单个数据库服务器的压力肯定也是越来越大,最终演变成数据库成为性能的瓶颈,而且当数据越来越多时,查询也更加耗费时间,当然数据库数据过大时,可以采用 ...
- SQL Server、MySQL主从搭建,EF Core读写分离代码实现
一.SQL Server的主从复制搭建 1.1.SQL Server主从复制结构图 SQL Server的主从通过发布订阅来实现 1.2.基于SQL Server2016实现主从 新建一个主库&quo ...
- MySQL集群(三)mysql-proxy搭建负载均衡与读写分离
前言 前面学习了主从复制和主主复制,接下来给大家分享一下怎么去使用mysql-proxy这个插件去配置MySQL集群中的负载均衡以及读写分离. 注意:这里比较坑的就是mysql-proxy一直没有更新 ...
随机推荐
- 【Linux】ssh设置了密钥,但ssh登陆的时候还需要输入密码
------------------------------------------------------------------------------------------------- | ...
- leetcode 473. 火柴拼正方形(DFS,回溯)
题目链接 473. 火柴拼正方形 题意 给定一串数,判断这串数字能不能拼接成为正方形 思路 DFS,但是不能每次从从序列开始往下搜索,因为这样无法做到四个边覆盖不同位置的值,比如输入是(5,5,5,5 ...
- 分布式系统:分布式任务调度xxl-job较深入使用
目录 系统关键概念介绍 执行器 任务 任务配置项描述 阻塞策略 路由策略 日志问题 客户端日志 服务端日志 框架目前发现的缺点以及存在的问题 xxl-job是一个分布式定时任务调度框架,功能强大,底层 ...
- Linux设置开机自动挂载镜像文件
1.将文件上传到服务器上(本例上传到/Data/software下) 2.挂载 mount -o loop /Data/software/rhel-server-7.6-x86_64-dvd.iso ...
- 使用gui_upload的总结
今天使用gui_upload函数将文本文件的内容读取到内表.出现了一个问题,总是程序宕掉,出项的提示是 Type conflict when calling a function module. 原来 ...
- Nifi组件脚本开发—ExecuteScript 使用指南(一)
Part 1 - 介绍 NiFi API 和 FlowFiles ExecuteScript 是一个万能的处理器,允许用户使用编程语言定义自己的数据处理功能, 在每一次 ExecuteScript p ...
- 忒修斯的Mac
我有一台Mac笔记本,用了快6年了,当初买它的时候还借了几千块. 三年前,它的屏幕坏了,修理的方式就是直接换屏,而换屏其实就是上半部分连壳带屏幕整个换掉,简单的说:另一台电脑的上半身嫁接过来. 今年, ...
- 。SLI,Service Level Indicator,服务等级指标,其实就是我们选择哪些指标来衡量我们的稳定性。而 SLO,Service Level Objective,服务等级目标,指的就是我们设定的稳定性目标,比如“几个 9”这样的目标。
.SLI,Service Level Indicator,服务等级指标,其实就是我们选择哪些指标来衡量我们的稳定性.而 SLO,Service Level Objective,服务等级目标,指的就是我 ...
- TCP报文段的首部格式 20字节的固定首部
<---- IP首部 TCP首部 TCP报文段的数据部分 <---- 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 ...
- 就是通过事件方法,在window.loaction.href里追加了参数字符串
参考博文:https://www.kancloud.cn/digest/yvettelau/137669