域名 AXUM.RS 将于2025年10月到期。我们无意再对其进行续费,我们希望你能够接续这个域名,让更多 AXUM 开发者继续受益。
  • 方案1️⃣AXUM.RS 域名 = 3000
  • 方案2️⃣方案1️⃣ + 本站所有专题原始 Markdown 文档 = 5000
  • 方案3️⃣方案2️⃣ + 本站原始数据库 = 5500
如果你有意接续这份 AXUM 情怀,请与我们取得联系。
说明:
  1. 如果有人购买 AXUM.RS 域名(方案1️⃣),或者该域名到期,本站将启用新的免费域名继续提供服务。
  2. 如果有人购买了 AXUM.RS 域名,且同时购买了内容和/或数据库(方案2️⃣/方案3️⃣),本站将关闭。届时我们或许会以另一种方式与你再相遇。

【实战】部署 MySQL 8 主从复制和读写分离

利用主从复制,可以实现 MySQL 的读写分离、热备份等。本章将带你实战 MySQL 的一主二从。

主从复制架构

实战环境

本次实战使用HAX的免费VPS的同一机房的三台VPS进行,操作系统为 Debian 11 x64。

节点IP
主节点(Master)2a01:4f8:13a:2690:face:10d4:*:0001
从节点1(Slave1)2a01:4f8:13a:2690:face:0915:*:0001
从节点2(Slave2)2a01:4f8:13a:2690:face:0e54:*:0001

安装MySQL

三个节点均需要安装 MySQL:

wget -O /tmp/mysql-apt-config.deb https://dev.mysql.com/get/mysql-apt-config_0.8.24-1_all.deb && \
dpkg -i /tmp/mysql-apt-config.deb && \
apt update -y && apt install mysql-server -y

安装过程中,需要一些交互:

选择要安装的组件

选择要安装的组件

这里保持默认,即安装 MySQL 8 Server。

设置 root 密码

设置root密码

这里输入 axum.rs 作为 root 用户的密码。注意,输入密码的时候可能没有回显,直接输入即可。输入完成按回车。

设置主节点(Master)

为了实现主从复制,我们需要对 MySQL 服务器进行设置。通过 apt 方式安装的 MySQL,它的 mysqld 配置文件位于 /etc/mysql/mysql.conf.d/mysqld.cnf

server-id = 1
read-only = 0
log-bin = axum-rs-bin
binlog-format = mixed
bind-address = ::

为简化操作,这里使用脚本来快速设置:

cat > /etc/mysql/mysql.conf.d/mysqld.cnf << EOF
[mysqld]
pid-file	= /var/run/mysqld/mysqld.pid
socket		= /var/run/mysqld/mysqld.sock
datadir		= /var/lib/mysql
log-error	= /var/log/mysql/error.log
server-id = 1
read-only = 0
log-bin = axum-rs-bin
binlog-format = mixed
bind-address = ::
EOF

设置完成后,重启 MySQL 服务,并尝试登录:

设置从节点(Slave)

从节点和主节点配置大同异,以 slave1 为例:

server-id = 2
read-only = 1
log-bin = axum-rs-bin
binlog-format = mixed
bind-address = ::
relay-log = axum-rs-relay
  • server-id:服务器ID,请确保在同一个主从复制中,其值的唯一性
  • read-only:对于从服务器来说,将其设置为只读是必要的。因为它的数据始终是由主节点同步而来。
  • relay-log:MySQL 8 新增的配置,用于开启和设置中继日志。

从节点1(slave1)

cat > /etc/mysql/mysql.conf.d/mysqld.cnf << EOF
[mysqld]
pid-file	= /var/run/mysqld/mysqld.pid
socket		= /var/run/mysqld/mysqld.sock
datadir		= /var/lib/mysql
log-error	= /var/log/mysql/error.log
server-id = 2
read-only = 1
log-bin = axum-rs-bin
binlog-format = mixed
bind-address = ::
relay-log = axum-rs-relay
EOF

systemctl restart mysql && \
mysql -u root -paxum.rs

从节点2(slave2)

cat > /etc/mysql/mysql.conf.d/mysqld.cnf << EOF
[mysqld]
pid-file	= /var/run/mysqld/mysqld.pid
socket		= /var/run/mysqld/mysqld.sock
datadir		= /var/lib/mysql
log-error	= /var/log/mysql/error.log
server-id = 3
read-only = 1
log-bin = axum-rs-bin
binlog-format = mixed
bind-address = ::
relay-log = axum-rs-relay
EOF

systemctl restart mysql && \
mysql -u root -paxum.rs

开启主从复制

主节点

创建主从复制的用户

CREATE USER 'slaver'@'%' IDENTIFIED BY 'axum.rs';
GRANT REPLICATION SLAVE ON *.* TO 'slaver'@'%';
ALTER USER 'slaver'@'%' IDENTIFIED WITH mysql_native_password BY 'axum.rs';
FLUSH PRIVILEGES;

查看主节点状态

SHOW MASTER STATUS;

/*
+--------------------+----------+--------------+------------------+-------------------+
| File               | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+--------------------+----------+--------------+------------------+-------------------+
| axum-rs-bin.000001 |     1147 |              |                  |                   |
+--------------------+----------+--------------+------------------+-------------------+
*/

请牢记两个值:

  • File:当前binlog 文件名
  • Position:当前位置

从节点

两个从节点都要执行以下操作。

设置同步数据的主节点信息

CHANGE MASTER TO 
MASTER_HOST='2a01:4f8:13a:2690:face:10d4:*:0001',
MASTER_USER='slaver',
MASTER_PASSWORD='axum.rs',
MASTER_SSL=0,
MASTER_LOG_FILE='axum-rs-bin.000001',
MASTER_LOG_POS=1147;
  • MASTER_HOST:主节点地址
  • MASTER_USER:执行同步的用户名
  • MASTER_PASSWORD:执行同步密码
  • MASTER_LOG_FILE:在“查看主节点状态”中,输出结果的 File 的值
  • MASTER_LOG_POS:在“查看主节点状态”中,输出结果的 Position

开启同步

START SLAVE;

查看同步状态

SHOW SLAVE STATUS\G

注意,只有当以下两个值均为 Yes时,才表示设置成功:

  • Slave_IO_Running
  • Slave_SQL_Running

从节点1的同步状态

从节点1的同步状态

从节点2的同步状态

从节点2的同步状态

测试主从同步

CREATE DATABASE axum_rs;
USE axum_rs;
CREATE TABLE axum_rs_test (
id INT UNSIGNED PRIMARY KEY AUTO_INCREMENT,
name VARCHAR(20) NOT NULL
);
INSERT INTO axum_rs_test (name) 
VALUES 
	('axum.rs'), ('test'), ('foobar');
SELECT * FROM axum_rs_test;

主节点创建并查询数据

从节点

SHOW DATABASES;
USE axum_rs;
SHOW TABLES;
SELECT * FROM axum_rs_test;

从节点1的查询结果

从节点2的查询结果

从节点2的查询结果

要查看完整内容,请先登录