主從介紹
Mysql主從又叫Replication、AB複製。簡單講就是A與B兩臺機器作主從後,在A上寫數據,另一臺B也會跟着寫數據,實現數據實時同步
mysql主從是基於binlog,主上需開啓binlog才能進行主從
主從過程大概有3個步驟
主將更改操做記錄到binlog裏
從將主的binlog事件(sql語句) 同步本機上並記錄在relaylog裏
從根據relaylog裏面的sql語句按順序執行python
mysql主從是異步複製過程
master開啓bin-log功能,日誌文件用於記錄數據庫的讀寫增刪
須要開啓3個線程,master IO線程,slave開啓 IO線程 SQL線程,
Slave 經過IO線程鏈接master,而且請求某個bin-log,position以後的內容。
MASTER服務器收到slave IO線程發來的日誌請求信息,io線程去將bin-log內容,position返回給slave IO線程。
slave服務器收到bin-log日誌內容,將bin-log日誌內容寫入relay-log中繼日誌,建立一個master.info的文件,該文件記錄了master ip 用戶名 密碼 master bin-log名稱,bin-log position。
slave端開啓SQL線程,實時監控relay-log日誌內容是否有更新,解析文件中的SQL語句,在slave數據庫中去執行。mysql
主從做用
實時災備,用於故障切換
讀寫分離,提供查詢服務
備份,避免影響業務linux
主從形式
一主一從
一主多從(擴展系統讀取的性能,由於讀是在從庫讀取的)
多主一從(5.7以後開始)
主主複製
聯機複製c++
主從複製步驟
主庫將全部的寫操做記錄在binlog日誌中,並生成log dump線程,將binlog日誌傳給從庫的I/O線程
從庫生成兩個線程,一個是I/O線程,另外一個是SQL線程
I/O線程去請求主庫的binlog日誌,並將binlog日誌中的文件寫入relay log(中繼日誌)中
SQL線程會讀取relay loy中的內容,並解析成具體的操做,來實現主從的操做一致,達到最終數據一致的目的sql
主從複製配置步驟
確保從數據庫與主數據庫裏的數據一致
在主數據庫裏建立一個同步帳戶受權給從數據庫使用
配合主數據庫(修改配置文件)
配置從數據庫(修改配置文件)數據庫
需求
搭建兩臺MYSQL服務器,一臺做爲主服務器,一臺做爲從服務器,主服務器進行寫操做,從服務器進行讀操做centos
環境說明
數據庫角色 IP 應用與系統 有無數據
主數據庫 192.168.4.219 centos7 mysql-5.7 有
從數據庫 192.168.4.218 centos7 mysql-5.7 無tomcat
在兩臺服務器上都按裝mysql
環境準備
關閉防火牆以SELINUX
service iptables stop
sed -ri 's/(SELINUX=).*/\1disabled/g' /etc/selinux/config
setenforce 0服務器
安裝mysql
安裝依賴包
yum -y install ncurses-devel openssl-devel openssl cmake mariadb-develdom
建立用戶和組
[root@yanyinglai ~]# groupadd -r -g 306 mysql
[root@yanyinglai ~]# useradd -M -s /sbin/nologin -g 306 -u 306 mysql
下載二進制格式的mysql軟件包
--[root@yanyinglai ~]# cd /usr/src/
--[root@yanyinglai src]#wget https://downloads.mysql.com/archives/get/file/mysql-5.7.22-linux-glibc2.12-x86_64.tar.gz
已下載
[root@localhost ~]# cd soft/
[root@localhost soft]# ls -ltrh
total 1.2G
-rw-r--r--. 1 root root 566M Jun 4 18:12 mysql-8.0.11-linux-glibc2.12-i686.tar.gz
-rw-r--r-- 1 root root 614M Jun 13 16:54 mysql-5.7.22-linux-glibc2.12-x86_64.tar.gz
解壓軟件至/usr/local/
[root@localhost soft]# tar xf mysql-5.7.22-linux-glibc2.12-x86_64.tar.gz -C /usr/local/
[root@localhost soft]# ls /usr/local/
bin config config1 config2 etc games include jdk1.6 lib lib64 libexec mysql-5.7.22-linux-glibc2.12-x86_64 python3 sbin share src tomcat
[root@localhost soft]# cd /usr/local/
[root@localhost local]# ln -sv mysql-5.7.22-linux-glibc2.12-x86_64/ mysql
`mysql' -> `mysql-5.7.22-linux-glibc2.12-x86_64/'
[root@localhost local]# ll
修改目錄/usr/locaal/mysql的屬主屬組
[root@localhost local]# chown -R mysql.mysql /usr/local/mysql
[root@localhost local]# ll /usr/local/mysql -d
lrwxrwxrwx 1 mysql mysql 36 Jun 20 17:17 /usr/local/mysql -> mysql-5.7.22-linux-glibc2.12-x86_64/
添加環境變量
[root@localhost local]# ls /usr/local/mysql
bin COPYING docs include lib man README share support-files
[root@localhost local]# cd
[root@localhost local]# echo 'export PATH=/usr/local/mysql/bin:$PATH' > /etc/profile.d/mysql.sh
[root@localhost local]# . /etc/profile.d/mysql.sh
[root@localhost local]# echo $PATH
/usr/local/mysql/bin:/usr/local/jdk1.6/bin:/usr/local/ssl/bin:/usr/local/mysql/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/root/bin
創建數據存放目錄
[root@localhost local]# cd /usr/local/mysql
[root@localhost mysql]# mkdir -p /opt/data
[root@localhost mysql]# chown -R mysql.mysql /opt/data/
[root@localhost mysql]# ll /opt/
total 0
drwxrwxr-x 7 mysql mysql 4096 Jun 18 15:25 data
初始化數據庫
[root@yanyinglai mysql]# /usr/local/mysql/bin/mysqld --initialize --user=mysql --datadir=/opt/data/
//這個命令的最後會生成一個臨時密碼,後面修改初始密碼須要
主庫218
2019-06-20T09:21:02.883424Z 1 [Note] A temporary password is generated for root@localhost: qij(Gork)2Nt
備庫219
2019-06-20T09:44:49.512519Z 1 [Note] A temporary password is generated for root@localhost: TU6w?erhvg7*
配置mysql
[root@localhost mysql]# ln -sv /usr/local/mysql/include/ /usr/local/include/mysql
`/usr/local/include/mysql' -> `/usr/local/mysql/include/'
[root@localhost mysql]# echo '/usr/local/mysql/lib' > /etc/ld.so.conf.d/mysql.conf
[root@yanyinglai ~]# ldconfig -v
生成配置文件,下面是顯示結果
[root@localhost mysql]# cat > /etc/my.cnf <<EOF
> [mysqld]
> basedir = /usr/local/mysql
> datadir = /opt/data
> socket = /tmp/mysql.sock
> port = 3306
> pid-file = /opt/data/mysql.pid
> user = mysql
> skip-name-resolve
> EOF
執行下面的
cat > /etc/my.cnf <<EOF
[mysqld]
basedir = /usr/local/mysql
datadir = /opt/data
socket = /tmp/mysql.sock
port = 3306
pid-file = /opt/data/mysql.pid
user = mysql
skip-name-resolve
EOF
配置服務啓動腳本
[root@localhost mysql]# cp -a /usr/local/mysql/support-files/mysql.server /etc/init.d/mysqld
[root@localhost mysql]# sed -ri 's#^(basedir=).*#\1/usr/local/mysql#g' /etc/init.d/mysqld
[root@localhost mysql]# sed -ri 's#^(datadir=).*#\1/opt/data#g' /etc/init.d/mysqld
啓動mysql
[root@localhost mysql]# service mysqld start
Starting MySQL.Logging to '/opt/data/localhost.localdomain.err'.
SUCCESS!
[root@localhost mysql]# ps -ef|grep mysql
root 4897 1 0 22:38 pts/2 00:00:00 /bin/sh /usr/local/mysql/bin/mysqld_safe --datadir=/opt/data --pid-file=/opt/data/mysql.pid
mysql 5075 4897 6 22:38 pts/2 00:00:01 /usr/local/mysql/bin/mysqld --basedir=/usr/local/mysql --datadir=/opt/data --plugin-dir=/usr/local/mysql/lib/plugin --user=mysql --log-error=yanyinglai.err --pid-file=/opt/data/mysql.pid --socket=/tmp/mysql.sock --port=3306
root 5109 4668 0 22:38 pts/2 00:00:00 grep --color=auto mysql
[root@localhost mysql]# ss -antl
State Recv-Q Send-Q Local Address:Port Peer Address:Port
LISTEN 0 128 *:22 *:*
LISTEN 0 100 127.0.0.1:25 *:*
LISTEN 0 128 :::22 :::*
LISTEN 0 100 ::1:25 :::*
LISTEN 0 80 :::3306
修改密碼
使用臨時密碼修改
[root@yanyinglai ~]# mysql -uroot -p
mysql> set password = password('123456');
mysql> quit
mysql主從配置
確保從數據庫與主數據庫的數據同樣先在主數據庫建立所須要同步的庫和表
[root@localhost mysql]# mysql -uroot -p123456
mysql> create database yan;
Query OK, 1 row affected (0.00 sec)
mysql> create database lisi;
Query OK, 1 row affected (0.00 sec)
mysql> create database wangwu;
Query OK, 1 row affected (0.00 sec)
mysql> use yan;
Database changed
mysql> create table tom (id int not null,name varchar(100)not null ,age tinyint);
Query OK, 0 rows affected (11.83 sec)
mysql> insert tom (id,name,age) values(1,'zhangshan',20),(2,'wangwu',7),(3,'lisi',23);
Query OK, 3 rows affected (0.07 sec)
Records: 3 Duplicates: 0 Warnings: 0
mysql> select * from tom;
備份主庫
備份主庫時須要另開一個終端,給數據庫上讀鎖,避免在備份期間有其餘人在寫入致使數據同步的不一致
[root@localhost mysql]# mysql -uroot -p123456
mysql> flush tables with read lock;
Query OK, 0 rows affected (0.01 sec)
//此鎖表的終端必須在備份完成之後才能退出(退出鎖表失效)
備份主庫並將備份文件傳送到從庫
[root@localhost ~]# mysqldump -uroot -p123456 --all-databases > /opt/all-20190620.sql
mysqldump: [Warning] Using a password on the command line interface can be insecure.
[root@localhost ~]# scp /opt/all-20190620.sql root@192.168.4.219:/opt/
解除主庫的鎖表狀態,直接退出交互式界面便可
mysql> quit
Bye
【同上步驟安裝從庫】
在從庫上恢復主庫的備份並查看是否與主庫的數據保持一致
安裝配置同上
[root@rdtest mysql]# mysql -uroot -p123456 < /opt/all-20190620.sql
mysql: [Warning] Using a password on the command line interface can be insecure.
[root@yanyinglai ~]# mysql -uroot -p123456
mysql> show databases;
mysql> use yan;
mysql> select * from tom;
主從配置
在主數據庫建立一個同步帳戶受權給從數據使用 IP對應爲slave的
[root@localhost ~]# mysql -uroot -p123456
mysql> create user 'repl'@'192.168.4.219' identified by '123456';
Query OK, 0 rows affected (5.50 sec)
mysql> grant replication slave on *.* to 'repl'@'192.168.4.219';
Query OK, 0 rows affected (0.04 sec)
mysql> flush privileges;
Query OK, 0 rows affected (0.09 sec)
配置主數據庫編輯配置文件
[root@localhost ~]# vi /etc/my.cnf
//添加如下內容
[mysqld]
basedir = /usr/local/mysql
datadir = /opt/data
socket = /tmp/mysql.sock
port = 3306
pid-file = /opt/data/mysql.pid
user = mysql
skip-name-resolve
//添加如下內容
//啓用binlog日誌
log-bin=mysql-bin
//主數據庫服務器惟一標識符 主的必須必從大
server-id=1
log-error=/opt/data/mysql.log
重啓mysql服務
[root@yanyinglai ~]# service mysqld restart
Shutting down MySQL..... SUCCESS!
Starting MySQL.Logging to '/opt/data/mysql.log'.
............................... SUCCESS!
[root@yanyinglai ~]# ss -antl
State Recv-Q Send-Q Local Address:Port Peer Address:Port
LISTEN 0 128 *:22 *:*
LISTEN 0 100 127.0.0.1:25 *:*
LISTEN 0 128 :::22 :::*
LISTEN 0 100 ::1:25 :::*
LISTEN 0 80 :::3306 :::*
[root@localhost ~]# mysql -uroot -p123456
查看主庫的狀態
mysql> show master status;
+------------------+----------+--------------+------------------+-------------------+
| File | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+------------------+----------+--------------+------------------+-------------------+
| mysql-bin.000003 | 154 | | | |
+------------------+----------+--------------+------------------+-------------------+
1 row in set (0.00 sec)
配置從數據庫
編輯配置文件
添加如下內容:
server-id=2 //設置從庫的惟一標識符 從的必須比主小
relay-log=mysql-relay-bin //啓用中繼日誌relay log
error-log=/opt/data/mysql.log
[root@rdtest mysql]# vi /etc/my.cnf
[mysqld]
basedir = /usr/local/mysql
datadir = /opt/data
socket = /tmp/mysql.sock
port = 3306
pid-file = /opt/data/mysql.pid
user = mysql
skip-name-resolve
server-id=2
relay-log=mysql-relay-bin
log-error=/opt/data/mysql.log
重啓從庫的mysql服務
[root@rdtest mysql]# service mysqld restart
Shutting down MySQL.. SUCCESS!
Starting MySQL.Logging to '/opt/data/mysql.log'.
SUCCESS!
[root@rdtest mysql]# ss -antl
State Recv-Q Send-Q Local Address:Port Peer Address:Port
LISTEN 0 128 :::22 :::*
LISTEN 0 128 *:22 *:*
LISTEN 0 100 ::1:25 :::*
LISTEN 0 100 127.0.0.1:25 *:*
LISTEN 0 10 *:5672 *:*
LISTEN 0 80 :::3306 :::*
[root@rdtest mysql]#
配置並啓動主從複製
主節點的二進制文件必定要是二進制列表中的最後一個二進制文件
mysql> change master to
-> master_host='192.168.4.218', //主庫IP
-> master_user='repl',//複製用戶
-> master_password='123456',//用戶密碼
-> master_log_file='mysql-bin.000003',
-> master_log_pos=154;
Query OK, 0 rows affected, 2 warnings (0.00 sec)
change master to
master_host='192.168.4.218',
master_user='repl',
master_password='123456',
master_log_file='mysql-bin.000003',
master_log_pos=154;
查看從服務器狀態
mysql> show slave status\G;
Slave_IO_Running: Yes //此處必須是yes
Slave_SQL_Running: Yes //此處必須是yes
ERROR:
No query specified
測試驗證在主服務器的yan庫的tom表插入數據:
mysql> use yan;
mysql> select * from tom;
mysql> insert tom(id,name,age) value (4,"zgy",18);
Query OK, 1 row affected (0.14 sec)
mysql> select * from tom;
insert tom(id,name,age) value (5,"zgy",16);
update tom set name='yyyyy' where age=18;
delete from tom where age=18;
在從數據庫查看是否數據同步
mysql> use yan;
mysql> select * from tom;
關注Slave_IO_State,Slave_IO_Running,Slave_SQL_Running狀態
若都爲yes狀態時確認同步配置完成
【檢查】
查看主庫二進制日誌是否開啓
mysql> show global variables like '%log%';
| log_bin | ON |
查看主節點二進制日誌列表
mysql> show master logs;
查看主節點的server id
mysql> show global variables like '%server%';
| server_id | 1 |
mysql> show global variables like '%log%';
| log_bin | OFF |
| relay_log | mysql-relay-bin |
| relay_log_basename | /opt/data/mysql-relay-bin |
| relay_log_index | /opt/data/mysql-relay-bin.index |
mysql> show global variables like '%server%';
| server_id | 2 |
從庫
開啓中繼日誌
添加:relay-log=relay-log
添加:relay-log-index=relay-log.index
添加:server-id=2
添加:innodb_file_per_table=ON
添加:skip_name_resolve=ON
查看進程
mysql> show processlist;
查看錶結構
mysql> show create table tom \G
*************************** 1. row ***************************
Table: tom
Create Table: CREATE TABLE `tom` (
`id` int(11) NOT NULL,
`name` varchar(100) NOT NULL,
`age` tinyint(4) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=latin1
1 row in set (0.00 sec)
執行計劃
mysql> explain select * from tom;
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
| 1 | SIMPLE | tom | NULL | ALL | NULL | NULL | NULL | NULL | 5 | 100.00 | NULL |
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------+
1 row in set, 1 warning (0.00 sec)
限制從服務器爲只讀
從服務器 read_only = ON,可是此限制對擁有SUPER權限的用戶均無效。
阻止全部用戶
mysq>FLUSH TABLES WITH READ LOCK;
【從庫只讀】
須要保證master-slave主從同步的salve庫,若是要設置爲只讀狀態,能夠限定普通用戶進行數據修改的操做,但不會限定具備super權限的用戶
set global read_only=1;
查詢有哪些數據庫
show databases;
查詢某數據庫有哪些表
use mysql
show tables;
獲取binlog文件列表
show binary logs;
只查看第一個binlog文件的內容
show binlog events;
查看指定binlog文件的內容
show binlog events in 'mysql-bin.000001';
初始化異常處理
[root@rdtest mysql]# ./bin/mysqld --initialize --user=mysql --datadir=/opt/data/
./bin/mysqld: error while loading shared libraries: libaio.so.1: cannot open shared object file: No such file or directory
安裝一下
[root@rdtest mysql]# yum install -y libaio
[root@rdtest mysql]# ./bin/mysqld --initialize --user=mysql --datadir=/opt/data/
2019-06-13T09:28:47.897255Z 1 [Note] A temporary password is generated for root@localhost: 4TB,9Q+&j3he
Error: Protected multilib versions: libstdc++-4.4.7-23.el6.i686 != libstdc++-4.4.7-17.el6.x86_64 You could try using --skip-broken to work around the problem You could try running: rpm -Va --nofiles --nodigest安裝下面的包yum --skip-broken updateyum clean allyum install libstdc++.so.6