zoukankan      html  css  js  c++  java
  • mysql主从复制

    mysql主从复制

    1.mysql复制简介

    复制是 MySQL的一项功能,允许服务器将更改从一个实例复制到另一个实例。
    • 主服务器将所有数据和结构更改记录到二进制日志中。
    • 从属服务器从主服务器请求该二进制日志并在本地应用其内容。

    2.mysql主从复制工作原理

    1.从库,从IO线程,读取master.info中的信息,获取数据库的连接参数(userpasswdipport),与上次请求过的binlog的位置(例子:mysql-bin.000003,position=640)
    2.IO线程连接主库,拿着binlog位置信息,问主库有没有比这个更新的二进制日志
    3.主库查询二进制日志,对比从库发过来的位置信息,如果有新的二进制日志,会通过dump thread发给从库
    4.从库通过IO线程,接收到主库发过来的二进制日志,会存储到TCP/IP缓存中,并返回ACK确认给主库,主库收到ACK之后就认变复制完成,继续其它工作
    5.从库会更新master.info文件,二进制日志更新到最新的位置
    6.从库IO线程会将TCP/IP缓存中的日志,存储到relaylog中继日志文件中
    7.从库SQL线程,读取relaylog.info,获取到上次执行到的relaylog日志位置,以这个位置为起点,往后继续执行中继日志
    8.从库SQL线程执行完所有relaylog后,会更新relaylog.inof信息为最新位置信息
    9.到此为之,一次完整的复制完成
    

    3.mysql主从复制搭建

    1、主从复制的前提

    1.1 两台以上mysql实例
       多台物理机
       多个mysql实例
    1.2 主库要开启二进制日志
    1.3 主库要提供复制相关的用户
        replication slave,一个比较特殊的权限
        grant replication slave on *.* to repl@'10.0.0.%' identified by '123';
    1.4 从库需要将和主库相差的数据,进行追加
        一般情况下可以人为备份主库数据,恢复到从库上
    1.5 从应该从恢复之后的时间点,开始自动从主库获取新的二进制日志开始应用
        我们需要人为告诉从库,从哪开始自动开始复制二进制日志(file+position),另外还需要告诉从库user,passwd,port,ip
        change master to
    

    2、复制中的线程及文件

    2.1、主库
    Dump(IO) thread:在复制过程中,主库发送二进制日志的线程
    2.2、从库
    IO thread:向主库请求二进制日志,并且接受二进制日志的线程
    SQL thread:执行请求过来的二进制的线程
    2.3、主库
    binlog文件:主库的二进制日志
    2.4、从库
    relaylog:中继日志,存储请求过来的二进制日志
    master.info:
        1、从库连接主库的重要参数(user,passwd,ip,port)
        2、上次获取过的主库二进制日志的位置
    relay-log.info
        存储从库SQL线程已经执行过的relaylog日志位置
    

    3.1 mysql主从复制搭建过程

    准备环境
    两个以上节点(多实例)
    3307:master
    3308:slave1
    3309:slave2

    第一步:各个节点配置,skip-name-resolve:关闭数据库的自动域名解析

    [root@db02 3307]# cat my.cnf 
    [mysqld]
    basedir=/application/mysql
    datadir=/data/3307
    server-id=3307
    port=3307
    log-bin=/data/3307/mysql-bin
    binlog_format=row
    socket=/data/3307/mysql.sock
    log-error=/data/3307/mysql.log
    skip-name-resolve
    [root@db02 data]# cat 3308/my.cnf 
    [mysqld]
    basedir=/application/mysql
    datadir=/data/3308
    server-id=3308
    port=3308
    socket=/data/3308/mysql.sock
    log-error=/data/3308/mysql.log
    skip-name-resolve
    [root@db02 data]# cat 3309/my.cnf 
    [mysqld]
    basedir=/application/mysql
    datadir=/data/3309
    server-id=3309
    port=3309
    socket=/data/3309/mysql.sock
    log-error=/data/3309/mysql.log
    skip-name-resolve
    

    第二步:启动多实例

    mysqld_safe --defaults-file=/data/3307/my.cnf &
    mysqld_safe --defaults-file=/data/3308/my.cnf &
    mysqld_safe --defaults-file=/data/3309/my.cnf &
    

    第三步:主库创建复制账户

    连接到主库:
    mysql -S /data/3307/mysql.sock
    grant replication slave on *.* to repl@'10.0.0.%' identified by '123';
    

    第四步:从库数据的追加
    (1)不需要追加的情况
    主和从同时搭建的新环境,就不需要备份主库数据,恢复到从库了,直接从第一个binlog(mysql-bin.000001)的开头位置(120)
    (2)如果主库已经工作了很长时间了,我们一般需要备份主库数据,恢复到从库,然后从库从备份的时间点起自动进行复制
    以下针对第二种方式:

    备份主库
    mysqldump -S /data/3307/mysql.sock -A -R  --triggers --master-data=2 --single-transaction >/backup/full.sql
    sed -n '22p' /backup/full.sql 
    -- CHANGE MASTER TO MASTER_LOG_FILE='mysql-bin.0000010', MASTER_LOG_POS=325
    恢复到从库
    mysql -S /data/3308/mysql.sock 
    mysql> set sql_log_bin=0;
    mysql> source /backup/full.sql
    

    第五步:从库开启主库

    mysql -S /data/3308/mysql.sock #连接到3308
    CHANGE MASTER TO
      MASTER_HOST='10.0.0.52', #主库数据库地址
      MASTER_USER='repl', #主库授权的账号
      MASTER_PASSWORD='123', #数据库密码
      MASTER_PORT=3307,  #主库的端口
      MASTER_LOG_FILE='mysql-bin.0000010', #主库的binlog文件
      MASTER_LOG_POS=325; #binlog位置
    开启主从(开启IO和SQL线程):
    start slave;
    查看主从状态:
    show slave statusG
    查看主从状态信息介绍
    Slave_IO_Running: Yes
    Slave_SQL_Running: Yes
    Last_IO_Errno: 0
    Last_IO_Error: 
    Last_SQL_Errno: 0
    Last_SQL_Error: 
    

    第六步:测试,在主数据库中新建数据库,在从库查看状态

    create databases oldoby;
    

    3.2 mysql主从复制故障解决方法

    3.2.1 IO线程故障

    1、主库连接不上
    user、password、port、ip 错误
    解决方案:
        stop  slave;    
        reset slave all;  
        change master to    
        start slave;
        防火墙
        网络不通
        skip-name-resolve
        stop  slave;
        start slave;
    2、主库二进制日志丢失或损坏
        解决方案:
        stop  slave;    
        reset slave all;  
        重新备份恢复
        change master to    
        start slave;
    

    3.2.2 SQL线程故障

    执行relaylog日志新事件
        1、删除、修改对象的操作时,没有这个对象
        2、创建对象时,对象已存在
        3、主键冲突
    从库做写入操作,会导致以上问题出现
    解决方法:
    stop slave; 
    set global sql_slave_skip_counter = 1; 
    start slave;
    /etc/my.cnf
    slave-skip-errors = 1032,1062,1007
    但是,以上操作有时是有风险的,最安全的做法就是重新构建主从。
    怎么预防以上问题?
    从库加入配置文件
    set global read_only=1;
    vim /etc/my.cnf
    read_only=1           ---->只能控制普通用户
    

    3.2.3 主从异常——主从延时过长

    show slave status G
    Seconds_Behind_Master:0 #查看同步时间
    默认的主从复制机制是异步的一个过程。
    主库原因:
    1、主库做修改操作之后,才会记录二进制日志。
    sync_binlog=0/1
        1:表示:每次事务commit,刷新binlog到磁盘
        0:系统决定binlog什时候刷新到磁盘
    2、主库的压力特别大(大事务、多事务)
    3、从库数量多,导致dump线程繁忙
    从库原因:
    1、relay-log写入慢
    2、SQL线程慢(主从硬件差异比较大)
    尽可能的避免主从延时
    1、sync_binlog=1
    2、大事务拆成小事务,多事务进行分离
    3、使用多级主从,分库分表架构
    4、将binlog放到ssd或者flash上,高性能存储
    5、将relay放到ssd或者flash上
    6、尽量选择和主库一致硬件和配置
    

    4.主从复制高级功能——半同步复制

    半同步:保证主从数据一致性的问题,安全的考虑
    5.5 出现的概念,但是不建议使用,性能太差
    5.6以后出现group commit 组提交功能,来提升开启版同步复制的性能
    5.7 增强半同步复制的新特性:after sync;
        加载插件
    主:
    INSTALL PLUGIN rpl_semi_sync_master SONAME 'semisync_master.so';
    从:
    INSTALL PLUGIN rpl_semi_sync_slave SONAME 'semisync_slave.so';
    查看是否加载成功:
    show plugins;
    启动:
    主:
    SET GLOBAL rpl_semi_sync_master_enabled = 1;
    从:
    SET GLOBAL rpl_semi_sync_slave_enabled = 1;
    重启从库上的IO线程
    STOP SLAVE IO_THREAD;
    START SLAVE IO_THREAD;
    查看是否在运行
    主:
    show status like 'Rpl_semi_sync_master_status';
    从:
    show status like 'Rpl_semi_sync_slave_status';
    查看同步时间
    show variables like '%rpl_sem%';
    +------------------------------------+-------+
    | Variable_name                      | Value |
    +------------------------------------+-------+
    | rpl_semi_sync_master_enabled       | ON    |
    | rpl_semi_sync_master_timeout       | 10000 |
    | rpl_semi_sync_master_trace_level   | 32    |
    | rpl_semi_sync_master_wait_no_slave | ON    |
    +------------------------------------+-------+
    4 rows in set (0.00 sec)
    默认情况先,到达10秒钟还没有ack,主从关系自动切换为普通复制
    如果是1主多从的半同步复制,只要有一台落地relaylog,返回ack,这次半同步就完成了。
    

    5.主从复制高级功能——延时从库

    为了防止逻辑损坏会专门找一个节点,配置成延时节点,一般情况下这个节点会被用备份

    设置延时,一般的情况下会设置3-6小时个延时时间,为会防止逻辑损坏
    mysql> stop slave;
    Query OK, 0 rows affected (0.00 sec)
    mysql> CHANGE MASTER TO MASTER_DELAY = 120; #设置延时时间
    Query OK, 0 rows affected (0.01 sec)
    mysql> start slave;
    Query OK, 0 rows affected (0.00 sec)
    mysql> show slave statusG
                        SQL_Delay: 120
    取消延时:
    mysql> stop slave;
    mysql> CHANGE MASTER TO MASTER_DELAY = 0;
    mysql> start slave;
    

    6.主从复制高级功能——复制过滤

    主要是为了不复制一些不需要的库或表到从库

    主库方面控制(不建议使用):
        白名单:只记录白名单中列出的库的二进制日志
         binlog-do-db
        黑名单:不记录黑名单列出的库的二进制日志
         binlog-ignore-db
    从库方面控制:
    白名单:只执行白名单中列出的库或者表的中继日志
    --replicate-do-db=test  #只复制test库
    --replicate-do-table=test.t1  #只复制test库下面的t1表
    --replicate-wild-do-table=test.x* #只复制test库下面以x开头的表
    黑名单:不执行黑名单中列出的库或者表的中继日志
    --replicate-ignore-db
    --replicate-ignore-table
    --replicate-wild-ignore-table
    

    例子:只同步oldboy库到从库

    在3308中的my.cnf配置文件中加入
    replicate-do-db=oldboy
    然后重启数据库
    mysqladmin -S /data/3308/mysql.sock shutdown
    mysql -S /data/3308/mysql.sock
    在主库上建数据库
    mysql> create database oldboy1;
    在从库上查看,发现没有同步过来
    mysql> show databases;
    +--------------------+
    | Database           |
    +--------------------+
    | information_schema |
    | mysql              |
    | oldboy             |
    | performance_schema |
    | test               |
    +--------------------+
    在主库上的oldboy进行添加数据
    mysql> use oldboy
    Reading table information for completion of table and column names
    You can turn off this feature to get a quicker startup with -A
    
    Database changed
    mysql> select * from t1;
    +------+------+
    | id   | name |
    +------+------+
    |    1 | li   |
    |    2 | aa   |
    |    3 | a1a  |
    +------+------+
    3 rows in set (0.00 sec)
    
    mysql> insert into t1 values(4,'12');
    Query OK, 1 row affected (0.00 sec)
    在从库上查看,发现已经同步过来了
    mysql> use oldboy
    Reading table information for completion of table and column names
    You can turn off this feature to get a quicker startup with -A
    
    Database changed
    mysql> select * from t1;
    +------+------+
    | id   | name |
    +------+------+
    |    1 | li   |
    |    2 | aa   |
    |    3 | a1a  |
    |    4 | 12   |
    +------+------+
    4 rows in set (0.00 sec)
    

    7.主从复制新特性-GTID复制

    7.1 /etc/my.cnf中添加配置

    gtid-mo                --启用gtid类型,否则就是普通的复制架构
    enforce-gtid-consistency=true        --强制GTID的一致性
    log-slave-updates=1            --slave更新是否记入日志
    

    7.2 GTID搭建过程

    三台机器:
    db01:172.16.1.51(master)
    db02:172.16.1.52(slave01)
    db02:172.16.1.53(slave02)

    7.2.1 编写配置文件

    1.编写master主机配置文件(172.16.1.51)
    [root@db01 ~]# cat /etc/my.cnf
    [mysqld]
    basedir=/application/mysql
    datadir=/application/mysql/data
    socket=/application/mysql/tmp/mysql.sock
    log-error=/var/log/mysql.log
    log_bin=/data/binlog/mysql-bin
    binlog_format=row
    skip-name-resolve
    server-id=51
    gtid-mode=on
    enforce-gtid-consistency=true
    log-slave-updates=1
    [client]
    socket=/application/mysql/tmp/mysql.sock
    [root@db01 ~]# 
    2.编写slave1主机配置文件(172.16.1.52)
    [root@db02 ~]# cat /etc/my.cnf
    [mysqld]
    basedir=/application/mysql
    datadir=/application/mysql/data
    server_id=52
    socket=/application/mysql/tmp/mysql.sock
    log-error=/var/log/mysql.log
    log_bin=/data/binlog/mysql-bin
    binlog_format=row
    skip-name-resolve
    gtid-mode=on
    enforce-gtid-consistency=true
    log-slave-updates=1
    [client]
    socket=/application/mysql/tmp/mysql.sock
    3.编写slave1主机配置文件(172.16.1.53)
    [root@db03 ~]# vim /etc/my.cnf
    [mysqld]
    basedir=/application/mysql
    datadir=/application/mysql/data
    server_id=53
    socket=/application/mysql/tmp/mysql.sock
    log-error=/var/log/mysql.log
    log_bin=/data/binlog/mysql-bin
    binlog_format=row
    skip-name-resolve
    gtid-mode=on
    enforce-gtid-consistency=true
    log-slave-updates=1
    [client]
    socket=/application/mysql/tmp/mysql.sock
    4.重启数据库
    /etc/init.d/mysqld restart
    

    7.2.2 配置复制环境

    1.在master主库上配置(51)
    grant replication slave  on *.* to repl@'172.16.1.%' identified by '123';
    2.在从库上配置(5152)
    change master to master_host='172.16.1.51',master_user='repl',master_password='123' ,MASTER_AUTO_POSITION=1;
    3.启动slave
    start slave;
    4.查看从库状态
    show slave statusG
    

    7.2.3 测试GTID环境

    1.在主库上新建数据库
    create database oldboy;
    2.在从库上查看有没有复制成功
    mysql> show databases;
    +--------------------+
    | Database           |
    +--------------------+
    | information_schema |
    | mysql              |
    | oldboy             |
    | performance_schema |
    | test               |
    +--------------------+
    5 rows in set (0.01 sec)
    
  • 相关阅读:
    MYSQL进阶学习笔记十:MySQL慢查询!(视频序号:进阶_23-24)
    MYSQL进阶学习笔记九:MySQL事务的应用!(视频序号:进阶_21-22)
    MYSQL学习拓展一:MySQL 存储过程之游标的使用!
    MYSQL进阶学习笔记八:MySQL MyISAM的表锁!(视频序号:进阶_18-20)
    linux初级学习笔记十:linux grep及正则表达式!(视频序号:04_4)
    linux初级学习笔记九:linux I/O管理,重定向及管道!(视频序号:04_3)
    MYSQL进阶学习笔记七:MySQL触发器的创建,应用及管理!(视频序号:进阶_16,17)
    linux shell 字符串操作
    How to Install JAVA 8 (JDK/JRE 8u111) on Debian 8 & 7 via PPA
    iptables Data filtering详解
  • 原文地址:https://www.cnblogs.com/yjiu1990/p/10845702.html
Copyright © 2011-2022 走看看