三高Mysql - 搭建“三高”架构之复制(下)

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
云数据库 RDS MySQL,高可用系列 2核4GB
简介: 三高Mysql - 搭建“三高”架构之复制(下)

GTID增强复制模式


经过上一个小节的介绍,我们发现传统主备节点复制的操作比较麻烦,特别是LOG_FILE + LOG_POS 的方式处理比较麻烦,根本原因是备库不知道从哪一个log开始进行复制,Mysql针对这一点在更高的版本中提供了全局事务的特性,给每一个事务配置一个唯一ID,也就是Mysql5.6的GTID增强模式,GTID就是 server_uuid:gno 组成一个键值对:

  • server_uuid(节点的UUID)
  • Gno:事务流水号(回滚之后进行回收)

启动GTID模式的配置很简单,在配置文件中加入如下的配置:

  • gitd_mode = on
  • enforece_gtid_consistency = on

最后使用GTID配置可以修改上一节最后部分提到的change..master部分:


change master to 
MASTER_HOST = 'xx.xx.xx.xx'
MASTER_USER = 'root'
master_auto_position = 1


GTID复制是为了增强主从复制减少故障率而出现的,推荐默认开启


binlog格式格式演变


注意在主从复制中最为关键的binlog格式随着Mysql的升级是做过调整的,在Mysql5.0之前的binlog格式是statement格式,同时内部记录的是原文,而对于一些特殊的语句来说同样的语句可能会有不同的效果,这时候就会有数据风险,比如下面的语句在处理对过程中会出现问题:


网络异常,图片无法展示
|


为什么要使用row格式


通过上面的介绍,我们大致了解了为什么需要使用row格式,因为row格式不记录SQL语句的原文,而是记录数据行的变化。但是row格式依然没有摆脱记录逻辑日志的这一条规则,而记录文本数据数据量比statement更大,之后会出现空间占用比较大的问题 。

进一步改进:mixed格式的binlog

针对binlog的row格式文本存储的问题,Mysql在提供了一种混合row和statement的对混合模式,对于有数据同步风险的使用row格式,而对于没有风险的则直接使用原文。

另外statement和row格式也称为给予语句的复制和给予行的复制,只是说法上的差别而已本质上并没有差别。


主备延迟如何处理


首先我们需要了解为什么主备之间存在延迟?

  • log的传送其实开销比较小,主要的消耗是消费relay log的耗时,
  • 备库的性能比主库要小很多
  • 备库承担了很多分析SQL的任务,压力比主库要打
  • 如果主库有长事务没有提交。

通常主备延迟有下面的解决方式:

  • 主备之间使用相同配置的机器
  • 备库关闭log实时落盘
  • 使用大数据系统分担日志处理任务

但是上面的处理也不是完美的,存在比较大的缺陷,并且通过上面的处理方式之后,依然没有办法完全排除所有的问题,还有诸如备库的性能由于被动接受复制,性能要比主库大打折扣,主库支持“多线程”而备库限制于“单线程”。

总结上面的内容可以发现主备延迟的特点如下:

  • 备库延迟主要是备库执行总是要比主库要慢。
  • 通过升级备库的硬件和关闭log实时落盘提高性能
  • 增加其他的组件分担复制的压力
  • 对于新时代的应用系统,使用“组复制”是官方的推荐选择。

针对上面的问题,Mysql对于传统的复制模式提供了更加细分的解决方式:并行复制。并行复制通常有两种思路,第一种是按表分发,第二种是按行分发,以及较新版本出现的事务组并行策略。

并行复制的原理

注意:Mysql 5.6的版本才出现并行复制。

并行复制是在主从复制同步的时候,从库在获取到主库的binlog日志并且保存为relay log之后,把重放relay log的任务另外分配一个叫做 worker的线程执行,sql线程则执行分配relay log的任务,从库只需要读取并分配任务不需要自己进行处理,从而更高效的重放relay log。

难点:如何分配relay log日志

这里还涉及几个关联问题:事务存在上下文依赖如何处理?如果存在冲突如何分配处理?(比如新增数据同时并行删除)。

并行复制名称看起来比较高大上,但是最大的问题是仅仅只是加快了重访relay log的速度,对于binlog 解析为relay log没有进行更多改进,也就是说把任务分担给了第三者让自己压力小了一点点,但是自己的处理速度和之前基本没太大变化。我们可以通过下面的图进行了解:


网络异常,图片无法展示
|


并行复制 - 分配思路:

因为并行复制的难点在于如何分发relay log,mysql提供了两种分配方式:按行分配按表分配

在最早期还有一种思路出现那就是按库并行的策略:这种处理方式是分发选择方式特别快,同时支持各种的log格式分发,但是同时缺点也非常明显,库粒度非常大并且负载均衡非常难。

开启方式:

salve-parallel-type=DATABASE,这是最初配置,由按照库的方式并行复制,这样的处理方式有下面的特点:

  • 分发选择非常快,支持各种log格式
  • 难以进行负载均衡,库粒度非常大。因为所有的worker实际上都是分配到一个单独的库进行处理,和之前的单线程处理方式并没有太大的却别。
    基于上面的一些问题,在后续的版本中Mysql对于并行复制进行了优化。

当然Mysql不会满足于库的粒度,所以后续基于按库复制基础上出现更多分配的方式:

按行分配:由于binlog记录的是数据行的改动内容,如果修改的不是同一行就可以分配,否则就把他们分配到同一个线程执行。

按表分配:语句按照不同的表进行分类,同一个表的事务放到同一个线程进行分配。

按事务组分配:Mysql5.7提出,使用事务组的方式进行并发提交和处理,下文将会单独介绍。


Mysql按照事务组并行策略(Mysql5.7新特性)

在介绍具体的策略之前,需要解释一下主库对于binlog刷盘的原理,binlog 刷盘分为两步动作,通过这两步动作之后,由此主库的binlog多线程访问,对于多线程事务的提交就可以进行并行刷盘的操作:

  1. binlog → binlog cache 写到binlog内存文件
  2. binlog内存文件刷到磁盘中(fsync 持久化磁盘)

事务组并行策略:


下面这图看起来十分复杂,其实简单理解可以认为每一次同步类似我们一次ctrl+s的动作,我们每一次的保存动作都需要刷新到磁盘,在多线程的操作过程中修改先是修改内存,然后按照顺序的进行刷盘。这里有读者可能会疑问,如果线程之间存在事务交叉怎么办?所以这里会依据binlog刷磁盘的逻辑,按照类似按行分配的处理方式将多线程写入到一个binlog缓冲文件之后一次刷新磁盘。

这样事务组并行缓冲合并刷新到方式,使得并行分配肯定会存在下面两种原则:

  • 能够在同一个组里提交的事务,一定不会修改同一行
  • 主库上可以并行执行的事务,备库上也一定是可以并行执行的

吐槽:其实这个特性说白了还是“抄”了Mysql原作者的思路,这里提一嘴MariaDB 是在 MySQL 版权被 Oracle 收购后,由 MySQL 创始人 Monty 创立的一个开源数据库,其版权授予了“MariaDB基金会(非营利性组织)。果然最了解儿子的还是亲爹。

下面的结构图是按照上面的文字描述的事物组并行策略的改进图:


网络异常,图片无法展示
|


可以看到上面的处理方式十分耗费IO性能,并行刷盘频繁浪费性能,可以发现最后一步可以合并前面的并行修改通过一次刷盘完成,所以出现了下面的优化方式:


网络异常,图片无法展示
|


事务组的含义就是将下面多个并行刷盘的操作合并为同一个,但是这时候又会有一个疑问到底等待多久合并并且刷新一次磁盘?

Mysql 使用了下面的参数进行控制:


(两个条件是或的关系)

  • binlog_group_commit_sync_delay:延迟多少微秒之后调用fsync()
  • binlog_group_commit_sync_no_dalay_count:类似多少次之后才调用fsync()

网络异常,图片无法展示
|


事务组并行策略优化(Mysql5.7.22版本)

在5.7版本中还存在过一个小版本的升级对于这个策略做了更多的扩展,比如下面的参数:

binlog-transaction-dependency-tracking 参数

  • COMMIT_ORDER:默认策略
  • WRITESET:没有修改相同行的事务可以并行。
  • WRITESET_SESSION:同一个线程先后执行两个事务不能并行。


强制走主库


如何判断备库已经追上去:

  • 强制延时
  • seconds_behind_master = 0
  • 对比binlog执行位点
  • 对比GTID对比情况

但是无法从根本上解决备库延迟的问题,它具备下面几个无法解决的根本性问题:

  • binlog 传送和redo log 重放需要时间,这时候受到网络IO或者磁盘IO阻塞的影响
  • 备库复制永远只能尽可能减小,无法从本质上完美解决延迟问题。
  • 备库因需要

针对次依然可以使用下面的方式判断具体事务是否重放:

  • 等待binlog位点:比如通过下面的命令直接监听到具体位置的变动,一旦有变动就认为主库的数据事务完成了。


网络异常,图片无法展示
|


  • 等待GTID(5.7.6之后每次都会返回GTID),通过下面的命令检查唯一事务ID:


网络异常,图片无法展示
|


简单-双主架构


主-主复制架构一半在一些项目比较小或者一些小公司经常使用,主主复制也就是两个库不存在主备关系,而是通过一个热备的库对于主节点宕机之后临时支撑业务使用。简单理解可以理解为柴油发电机,在停电的时候临时充当使用。

  • 两个节点均为Master
  • 两个节点均为Salve
  • 两库的数据互相复制
  • 如果其中一个出现问题,立刻切换到另一个库。


主主架构有下面一些比较明显的问题:

  • 数据冲突
  • 两边同时插入数据,出现冲突
  • 约定好插入不同ID
  • 只有一些库可写,另一个只读
  • 切换过快的数据库丢失问题。
  • 应用切换问题
  • 应用自己切换比较麻烦
  • keepalived 手段自动切换
  • 循环复制问题
  • 理论上的问题。
  • 未开GTID:使用ServerID过滤。


小结


在复制的部分介绍了复制的基本原理以及Mysql复制的三种方式,在5.6之后还提供了GTID的复制模式使得复制的故障率进一步下降,而针对复制的核心一方面是binlog文件,这里简单的介绍了binlog文件的三种写入格式,最早的statement,常用的row和一种推荐的混合模式,然而混合模式使用频率依然不如row多,内部的机制在实践过程中依然发现不少问题。

而另一方面则是在整个复制过程中插入“中间层”加快部分操作的处理速度,比如在重放relay log中加入一个worker专门负责处理和分发重放relay log的任务,Mysql在这个relay log重放分发的过程中做文章引入了并行复制,并行复制在早期使用按库的力度分配,这虽然很简单好实现但是因为粒度太大被立马改进,后续出现了按行分配和按表分配,最终出现了按事务组的策略分配,这些内容我们只需要理解,并不要去背诵或则牢记。

介绍完主备复制以及相关优化之后,我们切换视角来到了主库这边,主库这边也出现了更优秀的同步策略,那就是直接针对点进行监控,当然这种处理方式比较极端多数情况下不会接触到所以本部分没有过多介绍。

最后我们简单介绍了一下双主的架构,适合大部分的中小公司,对于个人开发的开源项目也能基本应付需求。


写在最后


复制部分仅仅是三高Mysql的第一个大关,后面和还有切换和扩展等着我们介绍,这里也会继续整理给大家带来更多好内容。

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助     相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
1月前
|
监控 关系型数据库 MySQL
深入了解MySQL主从复制:构建高效稳定的数据同步架构
深入了解MySQL主从复制:构建高效稳定的数据同步架构
99 1
|
2月前
|
NoSQL 关系型数据库 MySQL
微服务架构下的数据库选择:MySQL、PostgreSQL 还是 NoSQL?
在微服务架构中,数据库的选择至关重要。不同类型的数据库适用于不同的需求和场景。在本文章中,我们将深入探讨传统的关系型数据库(如 MySQL 和 PostgreSQL)与现代 NoSQL 数据库的优劣势,并分析在微服务架构下的最佳实践。
|
19小时前
|
存储 SQL 关系型数据库
Mysql高可用架构方案
本文阐述了Mysql高可用架构方案,介绍了 主从模式,MHA模式,MMM模式,MGR模式 方案的实现方式,没有哪个方案是完美的,开发人员在选择何种方案应用到项目中也没有标准答案,合适的才是最好的。
21 3
Mysql高可用架构方案
|
11天前
|
存储 关系型数据库 MySQL
MySQL主从复制原理和使用
本文介绍了MySQL主从复制的基本概念、原理及其实现方法,详细讲解了一主两从的架构设计,以及三种常见的复制模式(全同步、异步、半同步)的特点与适用场景。此外,文章还提供了Spring Boot环境下配置主从复制的具体代码示例,包括数据源配置、上下文切换、路由实现及切面编程等内容,帮助读者理解如何在实际项目中实现数据库的读写分离。
MySQL主从复制原理和使用
|
11天前
|
SQL 关系型数据库 MySQL
Mysql中搭建主从复制原理和配置
主从复制在数据库管理中广泛应用,主要优点包括提高性能、实现高可用性、数据备份及灾难恢复。通过读写分离、从服务器接管、实时备份和地理分布等机制,有效增强系统的稳定性和数据安全性。主从复制涉及I/O线程和SQL线程,前者负责日志传输,后者负责日志应用,确保数据同步。配置过程中需开启二进制日志、设置唯一服务器ID,并创建复制用户,通过CHANGE MASTER TO命令配置从服务器连接主服务器,实现数据同步。实验部分展示了如何在两台CentOS 7服务器上配置MySQL 5.7主从复制,包括关闭防火墙、配置静态IP、设置域名解析、配置主从服务器、启动复制及验证同步效果。
Mysql中搭建主从复制原理和配置
|
3月前
|
SQL 关系型数据库 MySQL
说一下MySQL主从复制的原理?
【8月更文挑战第24天】说一下MySQL主从复制的原理?
59 0
|
3月前
|
SQL 关系型数据库 MySQL
(二十五)MySQL主从实践篇:超详细版读写分离、双主热备架构搭建教学
在上篇《主从原理篇》中,基本上把主从复制原理、主从架构模式、数据同步方式、复制技术优化.....等各类细枝末节讲清楚了,本章则准备真正对聊到的几种主从模式落地实践,但实践的内容通常比较枯燥乏味,因为就是调整各种配置、设置各种参数等步骤。
482 2
|
3月前
|
SQL canal 关系型数据库
(二十四)全解MySQL之主从篇:死磕主从复制中数据同步原理与优化
兜兜转转,经过《全解MySQL专栏》前面二十多篇的内容讲解后,基本对MySQL单机模式下的各方面进阶知识做了详细阐述,同时在前面的《分库分表概念篇》、《分库分表隐患篇》两章中也首次提到了数据库的一些高可用方案,但前两章大多属于方法论,并未涵盖真正的实操过程。接下来的内容,会以目前这章作为分割点,开启MySQL高可用方案的落地实践分享的新章程!
1397 1
|
3天前
|
弹性计算 Kubernetes Cloud Native
云原生架构下的微服务设计原则与实践####
本文深入探讨了在云原生环境中,微服务架构的设计原则、关键技术及实践案例。通过剖析传统单体架构面临的挑战,引出微服务作为解决方案的优势,并详细阐述了微服务设计的几大核心原则:单一职责、独立部署、弹性伸缩和服务自治。文章还介绍了容器化技术、Kubernetes等云原生工具如何助力微服务的高效实施,并通过一个实际项目案例,展示了从服务拆分到持续集成/持续部署(CI/CD)流程的完整实现路径,为读者提供了宝贵的实践经验和启发。 ####
|
2天前
|
缓存 监控 API
探索微服务架构中的API网关模式
随着微服务架构的兴起,API网关成为管理和服务间交互的关键组件。本文通过在线零售公司的案例,探讨了API网关在路由管理、认证授权、限流缓存、日志监控和协议转换等方面的优势,并详细介绍了使用Kong实现API网关的具体步骤。
11 3

热门文章

最新文章