大型网站为了减轻服务器处理海量的并发访问,所产生的性能问题,采用了很多解决方案,其中最主流的解决方案就是读写分离,即将读操作和写操作分别导流到不同的服务器集群执行,到了数据业务层,数据访问层,如果还是传统的单靠一台服务器去扛,面对如此多的数据库连接操作,必然会崩溃,如果造成数据的丢失,是不堪设想的。本文将利用MySQL的主从配置,实现读写分离。
主从同步如何实现? 同步工作主要有三步
第一步就是主服务器(master)将对数据的操作记录到二进制日志文件(Binary log)中,也就是说,在每个事务更新数据完成之前,mster在日志里记录这些改变,mysql将事务串行地写入二进制文件中,在事件写入二进制日志完成后,master通知存储引擎提交事务,提交好事务后,就会进入第二步,需要补充一点的是,我们对数据的一次操作就称为一次二进制日志事件,也就是Binary log event。
第二步呢,Slave将master的Binary log拷贝到它的中继日志(Relay log)中,也就是Slave会首先开启一个工作线程(I/O thread),I/O线程在master上打开一个普通的连接,做Binary log的拷贝,从master的二进制日志文件中读取事件,如果已经跟上了master,就会睡眠并等待master产生新的事件,I/O线程然后将这些事件写入中继日志中;
第三步就是Slave重做中继日志事件,SQL线程从中继日志中读取事件,并重放其中的事件,更新Slave中的数据,使其与Master中的数据一致。
我在自己VMware虚拟机上装了两个centos系统,使用命令:# cat /etc/redhat-release 可以看到我自己的CentOs版本为7.6.1810的[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-w6z8vGXn-1595670172948)(https://upload-images.jianshu.io/upload_images/22459064-84c48acc091a0fd8.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)]
tar -zxvf mysql-5.7.31.tar.gz -C /usr/local
cd /usr/local
mv mysql-5.7.18-linux-glibc2.5-i686/ mysql
groupadd mysql
useradd -r -g mysql mysql
cd /usr/local/mysql
mkdir data
cd /usr/local
chown -R mysql:mysql mysql/
chmod -R 755 mysql/
cd /usr/local/mysql
./bin/mysqld --user=mysql --basedir=/usr/local/mysql --datadir=/usr/local/mysql/data --initialize
如果执行初始化时出现报错:./bin/mysqld: error while loading shared libraries: libnuma.so.1: cannot open shared object file: No such file or directory,就是少东西,centos使用yum装yum -y install numactl,装了后在执行初始化命令就可以了。
如果出现错误:initialize specified but the data directory has files in it. Aborting.大意是提示data目录有文件,把mysql安装目录data文件夹里的内容清空(建议读者备份到其他地方)。再次执行"mysqld --initialize",没有报错了!
记录下root的初始密码: Wg;iqpl_P5os
7. 将mysqld添加成服务,并启动它
建立mysql默认的配置文件/etc/my.cnf,并添加图片中内容
按esc,输入 :wq,保存并退出my.cnf,启动mysqld
查看mysql是否启动成功
出现如下信息表示启动成功
当然也可以查看mysqld的状态
8. 登录mysql并修改root密码
输入初始密码,步骤6中有生成,出现如下信息表示登录成功
修改root密码mysql> SET PASSword = PASSWORD(‘123456’);mysql> FLUSH PRIVILEGES;初次登录没有修改root的密码,操作数据库会出现如下错误提示,那么需要修改root用户的密码ERROR 1820 (HY000): You must reset your password using ALTER USER statement before executing this statement.
9. 设置mysql远程可访问(前提是防火墙必须关闭,chkconfig iptables off:设置自动启动为关闭,service iptables stop:关闭防火墙)先登录到mysqlmysql> use mysqlmysql> update user set host = ‘192.168.0.148’ where user = ‘root’;mysql> FLUSH PRIVILEGES;192.168.0.148即是可远程访问本地mysql的远程ip,若想任意ip都能访问本地mysql,那么只需要将192.168.0.148换成%即可mysql> update user set host = ‘%’ where user = ‘root’;
本文将ip为192.168.0.188的主机作为master,ip为192.168.0.152的主机作为slave,我用的mysql版本为:5.7.31。当然我们也可以在本机下载两套mysql软件,设置不同端口后进行启动,也是可以配置出主从同步来。1. master上配置bin-log日志修改mysql的配置文件:# vim /etc/my.cnf,加上如下内容,修改配置后需要重启mysql服务器
log-bin= master-bin #[必须]启用二进制日志
log-bin-index=master-bin.index
server-id = 4 #[必须]服务器唯一ID,默认是1,最好取ip的后3位
expire-logs-days = 7 #只保留7天的二进制日志,以防磁盘被日志占满
测试log_bin是否成功开启mysql> show variables like ‘%log_bin%’;出现下图,log_bin为ON则表示开启成功,OFF表示开启失败
配置好后,查看主服务器状态:show master status
2. slave上配置relay-log配置从服务器:vim /etc/my.cnf,点击 i 进入编辑模式,添加如下配置:
relay-log-index=slave-relay-bin.index
relay-log=slave-bin
server-id=152 #只要和master的server-id不一样就可以了
3. 将master和slave联系起来,slave需要知道master的地址
重启MySQL服务并设置读取锁定:登录mysql],mysql> flush tables with read lock;读取锁定的意思是只能读取,不能更新,以便获得一个一致性的快照查看主服务器上当前的二进制日志名和偏移量值:
4. 我们已经在主库配置好了允许从库访问主库的用户名和密码,下面我们对slave进行操作:mysql> change master to master_host=‘192.168.0.188’,master_port=3306,master_user=‘repl’,master_password=‘123’,master_log_file=‘master-bin.000001’,master_log_pos=154;其中的master_log_file,就是主库中file,在主库中使用 show master status;就可以看到了。
启动slave: mysql> start slave;查看slave从机的状态(G表示竖着展示): mysql> show slave status G
5. 关闭掉主数据库的读取锁定mysql> unlock tables;6. 测试 才开始的时候,我的主库和从库中的数据库是相同的,使用show databases查看,如下:
我现在主库执行创建库语句,执行:create database o2o; 我们去从库查询,会发现该数据库会自动同步到从库,如下:
通过测试验证,我们的mysql主从同步就配置好了,这里需要注意的是主库的mysql版本不能高于从库的mysql版本,因为MySQL总的语法是高版本兼容低版本的,如果从库的版本要比主库的版本低,可能主库执行的sql语句,从库执行就会报错,导致主从同步失败;从库只能读数据,不能在从库写数据,主库既可以读数据,也可以写数据。