`
codejm
  • 浏览: 30424 次
  • 来自: 青岛
社区版块
存档分类
最新评论

MySQL binlog_format (Mixed,Statement,Row)[转]

阅读更多

 

MySQL 5.5 中对于二进制日志 (binlog) 有 3 种不同的格式可选:Mixed,Statement,Row,默认格式是 Statement。总结一下这三种格式日志的优缺点。

MySQL Replication 复制可以是基于一条语句 (Statement Level) ,也可以是基于一条记录 (Row Level),可以在 MySQL 的配置参数中设定这个复制级别,不同复制级别的设置会影响到 Master 端的 bin-log 日志格式。

1. Row
日志中会记录成每一行数据被修改的形式,然后在 slave 端再对相同的数据进行修改。

优点:在 row 模式下,bin-log 中可以不记录执行的 SQL 语句的上下文相关的信息,仅仅只需要记录那一条记录被修改了,修改成什么样了。所以 row 的日志内容会非常清楚的记录下每一行数据修改的细节,非常容易理解。而且不会出现某些特定情况下的存储过程或 function ,以及 trigger 的调用和触发无法被正确复制的问题。

缺点:在 row 模式下,所有的执行的语句当记录到日志中的时候,都将以每行记录的修改来记录,这样可能会产生大量的日志内容,比如有这样一条 update 语句:

1
UPDATE product SET owner_member_id = 'b' WHERE owner_member_id = 'a'

执行之后,日志中记录的不是这条 update 语句所对应的事件 (MySQL 以事件的形式来记录 bin-log 日志) ,而是这条语句所更新的每一条记录的变化情况,这样就记录成很多条记录被更新的很多个事件。自然,bin-log 日志的量就会很大。尤其是当执行 alter table 之类的语句的时候,产生的日志量是惊人的。因为 MySQL 对于 alter table 之类的表结构变更语句的处理方式是整个表的每一条记录都需要变动,实际上就是重建了整个表。那么该表的每一条记录都会被记录到日志中。

2. Statement
每一条会修改数据的 SQL 都会记录到 master 的 bin-log 中。slave 在复制的时候 SQL 进程会解析成和原来 master 端执行过的相同的 SQL 再次执行。

优点:在 statement 模式下,首先就是解决了 row 模式的缺点,不需要记录每一行数据的变化,减少了 bin-log 日志量,节省 I/O 以及存储资源,提高性能。因为他只需要记录在 master 上所执行的语句的细节,以及执行语句时候的上下文的信息。

缺点:在 statement 模式下,由于他是记录的执行语句,所以,为了让这些语句在 slave 端也能正确执行,那么他还必须记录每条语句在执行的时候的一些相关信息,也就是上下文信息,以保证所有语句在 slave 端杯执行的时候能够得到和在 master 端执行时候相同的结果。另外就是,由于 MySQL 现在发展比较快,很多的新功能不断的加入,使 MySQL 的复制遇到了不小的挑战,自然复制的时候涉及到越复杂的内容,bug 也就越容易出现。在 statement 中,目前已经发现的就有不少情况会造成 MySQL 的复制出现问题,主要是修改数据的时候使用了某些特定的函数或者功能的时候会出现,比如:sleep() 函数在有些版本中就不能被正确复制,在存储过程中使用了 last_insert_id() 函数,可能会使 slave 和 master 上得到不一致的 id 等等。由于 row 是基于每一行来记录的变化,所以不会出现类似的问题。

3. Mixed
从官方文档中看到,之前的 MySQL 一直都只有基于 statement 的复制模式,直到 5.1.5 版本的 MySQL 才开始支持 row 复制。从 5.0 开始,MySQL 的复制已经解决了大量老版本中出现的无法正确复制的问题。但是由于存储过程的出现,给 MySQL Replication 又带来了更大的新挑战。另外,看到官方文档说,从 5.1.8 版本开始,MySQL 提供了除 Statement 和 Row 之外的第三种复制模式:Mixed,实际上就是前两种模式的结合。在 Mixed 模式下,MySQL 会根据执行的每一条具体的 SQL 语句来区分对待记录的日志形式,也就是在 statement 和 row 之间选择一种。新版本中的 statment 还是和以前一样,仅仅记录执行的语句。而新版本的 MySQL 中对 row 模式也被做了优化,并不是所有的修改都会以 row 模式来记录,比如遇到表结构变更的时候就会以 statement 模式来记录,如果 SQL 语句确实就是 update 或者 delete 等修改数据的语句,那么还是会记录所有行的变更。

其他参考信息

除以下几种情况外,在运行时可以动态改变 binlog 的格式
. 存储流程或者触发器中间;
. 启用了 NDB;
. 当前会话使用 row 模式,并且已打开了临时表;

如果 binlog 采用了 Mixed 模式,那么在以下几种情况下会自动将 binlog 的模式由 statement 模式变为 row 模式
. 当 DML 语句更新一个 NDB 表时;
. 当函数中包含 UUID() 时;
. 2 个及以上包含 AUTO_INCREMENT 字段的表被更新时;
. 执行 INSERT DELAYED 语句时;
. 用 UDF 时;
. 视图中必须要求运用 row 时,例如建立视图时使用了 UUID() 函数;

设定主从复制模式

1
2
3
4
log-bin=mysql-bin
#binlog_format="STATEMENT"
#binlog_format="ROW"
binlog_format="MIXED"

也可以在运行时动态修改 binlog 的格式。例如

1
2
3
4
5
6
mysql> SET SESSION binlog_format = 'STATEMENT';
mysql> SET SESSION binlog_format = 'ROW';
mysql> SET SESSION binlog_format = 'MIXED';
mysql> SET GLOBAL binlog_format = 'STATEMENT';
mysql> SET GLOBAL binlog_format = 'ROW';
mysql> SET GLOBAL binlog_format = 'MIXED';

两种模式的对比
Statement 优点
历史悠久,技术成熟;
产生的 binlog 文件较小;
binlog 中包含了所有数据库修改信息,可以据此来审核数据库的安全等情况;
binlog 可以用于实时的还原,而不仅仅用于复制;
主从版本可以不一样,从服务器版本可以比主服务器版本高;

Statement 缺点
不是所有的 UPDATE 语句都能被复制,尤其是包含不确定操作的时候;
调用具有不确定因素的 UDF 时复制也可能出现问题;
运用以下函数的语句也不能被复制:
* LOAD_FILE()
* UUID()
* USER()
* FOUND_ROWS()
* SYSDATE() (除非启动时启用了 –sysdate-is-now 选项)
INSERT … SELECT 会产生比 RBR 更多的行级锁;
复制须要执行全表扫描 (WHERE 语句中没有运用到索引) 的 UPDATE 时,须要比 row 请求更多的行级锁;
对于有 AUTO_INCREMENT 字段的 InnoDB 表而言,INSERT 语句会阻塞其他 INSERT 语句;
对于一些复杂的语句,在从服务器上的耗资源情况会更严重,而 row 模式下,只会对那个发生变化的记录产生影响;
存储函数(不是存储流程 )在被调用的同时也会执行一次 NOW() 函数,这个可以说是坏事也可能是好事;
确定了的 UDF 也须要在从服务器上执行;
数据表必须几乎和主服务器保持一致才行,否则可能会导致复制出错;
执行复杂语句如果出错的话,会消耗更多资源;

Row 优点
任何情况都可以被复制,这对复制来说是最安全可靠的;
和其他大多数数据库系统的复制技能一样;
多数情况下,从服务器上的表如果有主键的话,复制就会快了很多;
复制以下几种语句时的行锁更少:
* INSERT … SELECT
* 包含 AUTO_INCREMENT 字段的 INSERT
* 没有附带条件或者并没有修改很多记录的 UPDATE 或 DELETE 语句
执行 INSERT,UPDATE,DELETE 语句时锁更少;
从服务器上采用多线程来执行复制成为可能;

Row 缺点
生成的 binlog 日志体积大了很多;
复杂的回滚时 binlog 中会包含大量的数据;
主服务器上执行 UPDATE 语句时,所有发生变化的记录都会写到 binlog 中,而 statement 只会写一次,这会导致频繁发生 binlog 的写并发请求;
UDF 产生的大 BLOB 值会导致复制变慢;
不能从 binlog 中看到都复制了写什么语句(加密过的);
当在非事务表上执行一段堆积的 SQL 语句时,最好采用 statement 模式,否则很容易导致主从服务器的数据不一致情况发生;
另外,针对系统库 MySQL 里面的表发生变化时的处理准则如下:
如果是采用 INSERT,UPDATE,DELETE 直接操作表的情况,则日志格式根据 binlog_format 的设定而记录;
如果是采用 GRANT,REVOKE,SET PASSWORD 等管理语句来做的话,那么无论如何都要使用 statement 模式记录;
使用 statement 模式后,能处理很多原先出现的主键重复问题;

分享到:
评论

相关推荐

    mysql中binlog_format模式与配置详细分析

    `binlog_format` 参数定义了binlog的记录方式,有三种模式:STATEMENT、ROW和MIXED。 1. **STATEMENT模式(SBR)** - 在这种模式下,binlog记录的是修改数据的SQL语句。因为不需要记录每一行的变更,所以日志文件...

    mysql binlog的三种模式

    MySQL提供了三种不同的Binlog格式:Row Level、Statement Level 和 Mixed Level,每种格式都有其特点和适用场景。 #### 二、Row Level 模式 **Row Level** 是MySQL Binlog中的一种日志格式,它会记录每一行数据的...

    MySQL – binlog日志简介及设置

    MySQL binlog有三种格式:STATEMENT、ROW和MIXED。每种格式都有其特点: 1. **STATEMENT模式**:基于SQL语句的复制。在这种模式下,每条修改数据的SQL语句会被记录到binlog中。优点是日志文件较小,减少了磁盘I/O,...

    实例验证MySQL|update字段为相同的值是否会记录binlog|ysql-643108.pdf

    一 前言 前几天一个开发同事咨询我,update 更新字段为相同的值是否会记录binlog,我回复说不会。 其实 严格的说这个答案是不准确的,说要区分场景。...当 row_format 为mixed或者statement格式是,binlo

    binlog文件解析相关.docx

    本篇文章将深入探讨如何查看和解析MySQL的binlog文件,特别是针对两种不同的binlog_format模式——ROW和STATEMENT,以及与GTID(Global Transaction Identifier)的配合使用。 首先,我们可以使用`show master ...

    wing324#helloworld_zh#MySQL5-6之特别的RAND-函数1

    1.在没有特殊说明的情况下,默认binlog_format=STATEMENT,由于binlog_format=ROW/MIXED模式下,RAND()函数以行格

    mysql数据库恢复 binlog2sql数据库恢复

    这通常涉及到设置my.cnf配置文件,开启binlog功能,并设定合适的binlog_format(如ROW、STATEMENT或MIXED),以及binlog_row_image(FULL、MINIMAL或NO)。此外,还需要设置合适的binlog_size,以控制单个binlog文件...

    MySQL Flashback闪回功能详解.docx

    MySQL 的 binlog 有三种格式:statement、row、mixed。Statement 模式基于 SQL 语句的模式,一般来说生成的 binlog 尺寸较小,但是某些不确定性 SQL 语句或函数在复制过程可能导致数据不一致甚至出错;Row 模式基于...

    逐步分析MySQL从库com_insert无变化的原因

    在上述案例中,第一个业务的binlog_format设置为STATEMENT,这意味着主库的binlog包含了SQL语句。当这些语句在从库上重播时,com_insert计数器理应增加。然而,第二个业务使用的是ROW模式。在ROW模式下,binlog记录...

    MySQL日志的具体介绍

    * MIXED 混合了 STATEMENT 和 ROW 两种格式,默认采用 STATEMENT,在某些特殊情况下会自动切换为 ROW 进行记录。 show variables like '%binlog_format'; 变量 | 值 ---------------------+----------------------...

    24.MySQL是怎么保证主备一致的?1

    binlog有三种格式:statement、row和mixed。statement格式记录SQL语句本身,适合简单操作,但可能因依赖上下文导致主备不一致。例如,删除一行数据时,如果binlog_format设为statement,binlog会记录原始的delete...

    关于MySQL innodb_autoinc_lock_mode介绍

    在混合或行复制模式下(binlog_format=MIXED或ROW),这三个模式都是复制安全的。随着MySQL推荐将binlog_format设置为ROW,通常建议使用innodb_autoinc_lock_mode=2以获得更好的性能。 一个关于自动递增的例子展示...

    基于java开发的、高性能的、基于解析mysql row base binlog技术实现.zip

    Java开发的MySQL Row Base Binlog技术实现是一种高效的数据同步和备份解决方案,它通过解析MySQL的日志格式,特别是Row Format的binlog(二进制日志),来实现实时的数据复制和处理。以下是对这一技术的详细说明: ...

    MySql主从、双主同步文档

    - `binlog_format=mixed`:设置日志格式,默认为mixed,也可以设置为statement或row。 - `auto-increment-increment=2`:设置自动递增列的增量,适用于双主复制场景。 - `auto-increment-offset=1`:设置自动递增...

    MYSQL中binlog优化的一些思考汇总

    使用BINLOG_FORMAT和ROW/MIXED/STATEMENT模式 MySQL的binlog格式有ROW、STATEMENT和MIXED三种模式。ROW模式记录每个变更的具体行,不包含SQL语句,适合复杂的事务和多表操作;STATEMENT模式记录完整的SQL语句,但...

    mysql主从同步及canal配置详解.docx

    binlog 的格式可以是 Statement、Row 或 Mixed, Statement 格式记录的是 SQL 语句,Row 格式记录的是行级别的操作,Mixed 格式是 Statement 和 Row 的结合。 Canal 介绍 Canal 是 Alibaba 开源的一款 MySQL 数据...

    MySQL参数及启动选项详解.pdf

    `--binlog_format`则定义了二进制日志的格式,包括基于行的(row)、基于语句的(statement)和混合的(mixed)模式,每种模式有其适用场景。 `--binlog_cache_size`设定在事务期间用于存储二进制日志SQL语句的缓存大小,...

    mysql全局变量详解[收集].pdf

    10. **binlog_format**:MySQL复制有三种主要方式:基于语句的复制(SBR)、基于行的复制(RBR)和混合模式复制(MBR)。不同的模式会影响binlog的记录方式和可能产生的复制问题。SBR节省空间但可能导致数据不一致,...

    MySQL数据库误操作后快速回滚的方法

    如果binlog_format是STATEMENT或MIXED,那么可能无法精确恢复,因为这些模式下binlog只记录SQL语句,而不包含具体影响的行数据。 此外,为了防止类似问题的发生,定期备份数据库和binlog也是必要的。你可以使用...

Global site tag (gtag.js) - Google Analytics