日志是 MySQL 数据库的重要组成部分,比如数据持久化、主从复制、数据回滚等操作都依赖日志系统来实现。本文将介绍MySQL的三种日志:归档日志binlog、重做日志redo log 和回滚日志undo log。
二进制日志(binary log)描述了数据库更改的“事件”,比如建表、更改表数据等操作,它属于逻辑日志,由 Server
层记录,记录了数据库所有的逻辑操作。还包含了关于每条语句更新数据的时间信息,不记录查询语句的日志,比如SELECT语句,如果要查看所有语句的日志,可以使用通用查询日志(general query log)。
binlog主要有两个用途:
下面来看看MySQL中如何使用binlog。
查看binlog是否开启:show variables like '%log_bin%';
mysql> show variables like '%log_bin%';
ERROR 2006 (HY000): MySQL server has gone away
No connection. Trying to reconnect...
Connection id: 15
Current database: mysql
+---------------------------------+------------------------------------------------+
| Variable_name | Value |
+---------------------------------+------------------------------------------------+
| log_bin | ON |
| log_bin_basename | D:\tools\mysql-8.0.16-winx64\data\binlog |
| log_bin_index | D:\tools\mysql-8.0.16-winx64\data\binlog.index |
| log_bin_trust_function_creators | OFF |
| log_bin_use_v1_row_events | OFF |
| sql_log_bin | ON |
+---------------------------------+------------------------------------------------+
6 rows in set, 1 warning (0.81 sec)
如果没有开启,需修改配置文件,新增:
[mysqld]
log-bin=mysql-bin
然后重启MySQL。
MySQL 5.7.7 之后的版本中,binlog日志格式默认采用ROW
,Row 格式不记录 SQL 语句上下文相关信息,仅记录每一行的数据修改。
mysql> show variables like 'binlog_format';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| binlog_format | ROW |
+---------------+-------+
1 row in set, 1 warning (0.00 sec)
binlog日志通常在事务提交时才记录,mysql 通过 sync_binlog
参数来控制 biglog 刷入磁盘的时机,其取值范围是 0-N
:
MySQL 5.7.7 之后的版本默认为 1
。
mysql> show variables like 'sync_binlog';
+---------------+-------+
| Variable_name | Value |
+---------------+-------+
| sync_binlog | 1 |
+---------------+-------+
1 row in set, 1 warning (0.00 sec)
刷新日志:Flush logs;
, 会产生一个新编号的binlog文件。
查看所有binlog文件:
mysql> show binary logs;
+---------------+-----------+-----------+
| Log_name | File_size | Encrypted |
+---------------+-----------+-----------+
| binlog.000742 | 178 | No |
| binlog.000743 | 1571 | No |
| binlog.000744 | 415 | No |
| binlog.000745 | 155 | No |
+---------------+-----------+-----------+
4 rows in set (0.00 sec)
查看当前写入的binlog文件:
mysql> show master status;
+---------------+----------+--------------+------------------+-------------------+
| File | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+---------------+----------+--------------+------------------+-------------------+
| binlog.000745 | 155 | | | |
+---------------+----------+--------------+------------------+-------------------+
1 row in set (0.00 sec)
接下来创建一个数据表,并插入数据库,SQL语句如下:
create table department(
id int,
name varchar(255) not null
);
ALTER TABLE department CHANGE COLUMN name dept varchar(255);
insert into department (id, dept) values (1,'开发');
insert into department (id, dept) values (2,'测试');
update department set dept=产品 where id=2;
查看数据:
mysql> select * from department;
+------+------+
| id | dept |
+------+------+
| 1 | 开发 |
| 2 | 测试 |
+------+------+
2 rows in set (0.00 sec)
然后使用 show binlog events;
命令查看binlog内容:
mysql> show binlog events in 'binlog.000745';
+---------------+------+----------------+-----------+-------------+---------------------------------------------------------------------------------------------------+
| Log_name | Pos | Event_type | Server_id | End_log_pos | Info |
+---------------+------+----------------+-----------+-------------+---------------------------------------------------------------------------------------------------+
| binlog.000745 | 4 | Format_desc | 1 | 124 | Server ver: 8.0.16, Binlog ver: 4 |
| binlog.000745 | 124 | Previous_gtids | 1 | 155 | |
| binlog.000745 | 155 | Anonymous_Gtid | 1 | 232 | SET @@SESSION.GTID_NEXT= 'ANONYMOUS' |
| binlog.000745 | 232 | Query | 1 | 393 | use `testdb`; create table department(
id int,
name varchar(255) not null
) /* xid=191 */ |
| binlog.000745 | 393 | Anonymous_Gtid | 1 | 470 | SET @@SESSION.GTID_NEXT= 'ANONYMOUS' |
| binlog.000745 | 470 | Query | 1 | 621 | use `testdb`; ALTER TABLE department CHANGE COLUMN name dept varchar(255) /* xid=192 */ |
| binlog.000745 | 621 | Anonymous_Gtid | 1 | 700 | SET @@SESSION.GTID_NEXT= 'ANONYMOUS' |
| binlog.000745 | 700 | Query | 1 | 777 | BEGIN |
| binlog.000745 | 777 | Table_map | 1 | 843 | table_id: 100 (testdb.department) |
| binlog.000745 | 843 | Write_rows | 1 | 891 | table_id: 100 flags: STMT_END_F |
| binlog.000745 | 891 | Xid | 1 | 922 | COMMIT /* xid=193 */ |
| binlog.000745 | 922 | Anonymous_Gtid | 1 | 1001 | SET @@SESSION.GTID_NEXT= 'ANONYMOUS' |
| binlog.000745 | 1001 | Query | 1 | 1078 | BEGIN |
| binlog.000745 | 1078 | Table_map | 1 | 1144 | table_id: 100 (testdb.department) |
| binlog.000745 | 1144 | Write_rows | 1 | 1192 | table_id: 100 flags: STMT_END_F |
| binlog.000745 | 1192 | Xid | 1 | 1223 | COMMIT /* xid=194 */ |
+---------------+------+----------------+-----------+-------------+---------------------------------------------------------------------------------------------------+
16 rows in set (0.00 sec)
可以看到binlog记录了执行过程。
先删除数据表:
mysql> drop table department;
Query OK, 0 rows affected (0.28 sec)
mysql> select * from department;
ERROR 1146 (42S02): Table 'testdb.department' doesn't exist
windows中可以使用 mysqlbinlog.exe 来使用binlog恢复数据到某个位置,我的存放地址为 D:\tools\mysql-8.0.16-winx64\bin
。
使用mysqlbinlog命令恢复:
$ mysqlbinlog --no-defaults "D:\tools\mysql-8.0.16-winx64\data\binlog.000745" --start-position=124 --stop-position=1223 > d:\\test.sql
登录mysql服务器,执行:source d:\\test.sql
查询数据:
mysql> select * from department;
+------+------+
| id | dept |
+------+------+
| 1 | 开发 |
| 2 | 测试 |
+------+------+
2 rows in set (0.00 sec)
数据成功恢复了。
前面介绍了binlog日志是在事务提交时才记录的,如果MySQL在事务执行过程中突然奔溃,使用binlog日志无法保证事务完整性,没有 crash-safe 的能力,这也是MySQL 引擎 MyISAM的一个缺点。crash-safe指MySQL服务器奔溃重启后,能够保证:
InnoDB 引擎通过redo log(重做日志)和undo log(回滚日志)这两个日志保证了MySQL的crash-safe 能力,他们都是InnoDB 引擎引入的日志,属于引擎层的日志。前面说了binlog是server层记录的日志,所以使用各种引擎的MySQL服务器都可以使用它。
redo log属于物理日志,记录的是事务对某个数据页做了哪些修改,只记录修改相比存储整个数据页的性能更优。redo log的记录采用了WAL(Write-Ahead Logging) 技术,也就是先写日志,再写磁盘,这样进一步提升了性能。
当执行一条DML语句的时候,InnoDB 引擎会先把记录写到内存(redo log buffer)中,然后在系统在适当的时候将这个操作记录更新到磁盘(redo log file)。这个写入磁盘的时机可以通过 innodb_flush_log_at_trx_commit
参数来配置:
mysql> show variables like 'innodb_flush_log_at_trx_commit';
+--------------------------------+-------+
| Variable_name | Value |
+--------------------------------+-------+
| innodb_flush_log_at_trx_commit | 1 |
+--------------------------------+-------+
1 row in set, 1 warning (0.00 sec)
redo log大小固定,通过循环写入的方式,
mysql> show variables like '%innodb_log%';
+------------------------------------+----------+
| Variable_name | Value |
+------------------------------------+----------+
| innodb_log_buffer_size | 16777216 |
| innodb_log_checksums | ON |
| innodb_log_compressed_pages | ON |
| innodb_log_file_size | 50331648 |
| innodb_log_files_in_group | 2 |
| innodb_log_group_home_dir | .\ |
| innodb_log_spin_cpu_abs_lwm | 80 |
| innodb_log_spin_cpu_pct_hwm | 50 |
| innodb_log_wait_for_flush_spin_hwm | 400 |
| innodb_log_write_ahead_size | 8192 |
+------------------------------------+----------+
10 rows in set, 1 warning (0.01 sec)
redo log 的写入包括了两个步骤:prepare 和 commit,也就是"两阶段提交"。目的是为了使 redo log 和 binlog 保持一致,步骤如下:
使用两阶段提交可以保证主从数据一致。
redo log保证数据持久性,而undo log用于保证事务操作的原子性。undo log属于逻辑日志,记录了数据的反向操作语句,保证进行事务回滚时,可以将数据还原。主要有两个功能:
mysql> show variables like '%undo%';
+--------------------------+------------+
| Variable_name | Value |
+--------------------------+------------+
| innodb_max_undo_log_size | 1073741824 |
| innodb_undo_directory | .\ |
| innodb_undo_log_encrypt | OFF |
| innodb_undo_log_truncate | ON |
| innodb_undo_tablespaces | 2 |
+--------------------------+------------+
5 rows in set, 1 warning (0.01 sec)
innodb_max_undo_log_size
:日志文件大小innodb_undo_directory
:存放目录innodb_undo_tablespaces
:创建undo文件个数,会在mysql的data目录下创建命名为undo_001~undo_002的undo tablespace文件。undo log默认存放在 data/ibdata1
中,如果开启了 innodb_file_per_table
则会保存到每个表的.ibd文件中。
mysql> show variables like 'innodb_file_per_table';
+-----------------------+-------+
| Variable_name | Value |
+-----------------------+-------+
| innodb_file_per_table | ON |
+-----------------------+-------+
1 row in set, 1 warning (0.01 sec)
简单介绍了这3种日志之后,来看一下一条更新语句执行过程中这些日志是如何记录的。
update department set dept='产品' where id=2;
主要流程如下:
dept
设置为 产品
。本文介绍了MySQL的三种重要日志:物理日志 redo log和undo log 和逻辑日志 binlog,它还有其它日志,比如Error log、General query log、Relay log、Slow query log、DDL log等,他们都是MySQL数据库的重要组成部分,日志记录在一定程度上影响MySQL服务的性能,但他们是必不可少的,备份的频率需要根据具体业务来进行配置。
他们只要拒绝相信就行。面对难题,最容易的对策就是拒绝相信它的存在。——艾萨克·阿西莫夫《神们自己》