本节列出和解释了组复制相关的要求和限制。html
要使用组复制,每一个MySQL节点必须知足如下条件:mysql
组中的每一个成员都必须配置如下选项:sql
--log-bin[=log_file_name]
。MySQL组复制会复制二进制日志的内容,所以必须开启二进制日志。--log-slave-updates
。节点须要记录applier已应用的日志。组中的每一个节点都须要记录它们所接收到并应用的全部事务,这是必须的,由于恢复过程是依赖于组中参与者的二进制日志来进行的。所以,组中每一个成员都必须保留每一个事务的副本,即便某事务不是在该节点上开始的。--binlog-format=row
。组复制依赖于基于行格式的二进制日志,以便在组中传播所发生的更改能保持一致性。并且,在探测组中不一样节点间发生的并发事务是否冲突时,须要从行格式的日志中提取一些内容来作比较。--gtid-mode=ON
。组复制使用GTID(全局事务ID)来精确跟踪每一个节点上已经提交了哪些事务。也所以能够推断出某节点上要执行的事务是否和已执行的事务(每一个节点上都有副本)冲突。换句话说,GTID是整个组复制判断事务是否冲突的基础。--master-info-repository=TABLE
和--relay-log-info-repository=TABLE
。applier须要将 master 和 relay log 的元数据信息写入到系统表 mysql.slave_master_info 和 mysql.slave_relay_log_info 中。这保证了组复制插件具备一致性恢复的能力和复制的元数据事务管理能力。--transaction-write-setextraction=XXHASH64
,以便将行写入到二进制日志中时,节点也收集写集。写集基于每行的主键,是惟一标识被更改行的标签的简化形式,该标签后续会用于探测事务冲突性。--slave-parallel-workers=N
(N是applier线程数量)、--slavepreserve-commit-order=1
以及--slave-parallel-type=LOGICAL_CLOCK
。--slaveparallel-workers=N
表示启用多applier线程,组复制依赖于创建在全部参与节点都以相同顺序接收和应用、提交事务的一致性机制,所以还必须设置--slave-preserve-commit-order=1
以保证并行事务的最终提交是和原事务所在顺序位置一致的。最后,为了决定哪些事务能够并行执行,relay log 必须包含由--slave-parallel-ype=LOGICAL_CLOCK
生成的事务父信息(transaction parent information)。当尝试加入一个只设置了--slave-parallel-workers
大于0,却没有设置其余两项的新成员,将会报错并阻止它的加入。下面是使用组复制已知的限制:网络
--binlog-checksum=NONE
。Gap Locks:在验证阶段中(certification process),不会考虑 Gap Locks,所以在 InnoDB 的外部没法获取任何关于Gap 锁的信息。多线程
注意:并发
除非你的应用程序或业务需求依赖于REPEATABLE READ(MySQL默认该隔离级别),不然建议在组复制中使用READ COMMITTED隔离级别。在READ COMMITTED隔离级别中,InnoDB基本上不会使用Gap Locks,这将使得InnoDB自带的冲突探测能和组复制的冲突探测相互对齐从而保持一致。app
group_replication_single_primary_mode=OFF
)不支持多级外键依赖,特别是表上定义了级联的外键约束(CASCADING foreign key constraints)。这是由于多主模型下执行外键约束的级联操做可能会出现未检测到的冲突,从而致使组内成员间数据不一致。所以,咱们推荐在使用多主模型时,在每一个节点上都设置group_replication_enforce_update_everywhere_checks=ON
以免出现未检测到的冲突。在单主模型下没有这种问题,由于没有并发写操做,从而不可能会出现未被探测到的冲突。LOAD DATA INFILE
的文件切割为多个小块。多主模型可能出现死锁:在多主模型下,SELECT ... FOR UPDATE
语句可能会致使死锁。这是由于组内成员之间不会共享锁资源(译注:share nothing),所以这样的语句可能达不到预期的结果。性能