僅供自學使用。css
MySQL屬於傳統關係型數據庫產品,它開放式的架構使得用戶選擇性很強,同時社區開發與維護人數衆多。其功能穩定,性能卓越,且在遵照GPL協議的前提下,能夠無償使用與修改,也爲MySQL的推廣與使用帶來了更多的利好。在MySQL成長與發展過程當中,支持的功能逐漸增多,性能也不斷提升,對平臺的支持也愈來愈多。
MySQL是一種關係型數據庫管理系統,關係型數據庫的特色是將數據保存在不一樣的表中,再將這些表放入不一樣的數據庫中,而不是將全部數據統一放在一個大倉庫裏,這樣的設計增長了MySQL的讀取速度,並且靈活性和可管理性也獲得了很大提升。訪問及管理MySQL數據庫的最經常使用標準化語言爲SQL結構化查詢語言。html
自甲骨文公司收購MySQL後,其在商業數據庫與開源數據庫領域市場的佔有份額都躍居第一,這樣的格局引發了業內不少的人士的擔心,由於商業數據庫的老大有可能將MySQL閉源。爲了不Oracle將MySQL閉源,而無開源的類MySQL數據庫可用,MySQL社區採用分支的方式來避開這個風險。MariaDB數據庫就這樣誕生了,MariaDB是一個向後兼容,可能在之後替代MySQL的數據庫產品,其官方地址爲:https://mariadb.org/ 。不過,這裏仍是建議你們選擇更穩定,使用更普遍的MySQL數據庫,能夠先測試MariaDB數據庫,等使用的人員更多一些,社區更活躍後再考慮使用爲好。前端
在以前LNMP的講解中,已經針對MySQL數據庫進行了介紹,並說明了爲何要選擇MySQL數據庫,以及MySQL數據庫在Linux系統下的多種安裝方式,同時講解了MySQL的二進制方式單實例安裝,基礎優化等內容,本節將爲同窗們講解更爲實用的MySQL多實例安裝,主從複製集羣等重要應用實踐。java
MySQL多實例的做用以下:mysql
(1)有效利用服務器資源面試
當單個服務器資源有剩餘時,能夠充分利用剩餘的資源提供更多的服務,且能夠實現資源的邏輯隔離。sql
(2)節約服務器資源shell
當公司資金緊張,可是數據庫又須要各自盡可能獨立地提供服務,並且,須要主從複製等技術時,多實例就再好不過了。數據庫
MySQL多實例有它的好處,但也有其弊端,好比,會存在資源互相搶佔的問題。vim
當某個數據庫實例併發很高或有SQL慢查詢時,整個實例會消耗大量的系統CPU,磁盤I/O等資源,致使服務器上的其餘數據庫實例提供服務的質量一塊兒降低。這就至關於你們住在一個房子的不一樣臥室同樣,早晨起來上班,都要刷牙,洗臉等,這樣衛生間就會長期佔用,其餘人要等待同樣。不一樣實例獲取的資源是相對獨立的,沒法像虛擬化同樣徹底隔離。
若公司資金緊張,公司業務訪問量不太大,但又但願不一樣業務的數據庫服務各自盡可能獨立地提供服務而互相不受影響,同時,還須要主從複製等技術提供備份或讀寫分離服務,那麼多實例就再好不過了。例如:能夠經過3臺服務器部署9~15個實例,交叉作主從複製,數據備份及讀寫分離,這樣就可達到9~15臺服務器每一個只裝一個數據庫纔有的效果。這裏要強調的是,所謂的儘可能獨立是相對的。
當公司業務訪問量不太大的時候,服務器的資源基本上都浪費了,這時就很適合多實例的應用,若是對SQL語句的優化作得比較好,MySQL多實例會是一個很值得使用的技術,即便併發很大,合理分配好系統資源,搭配好服務,也不會有太大問題。
門戶網站一般都會使用多實例,由於配置硬件好的服務器,可節省IDC機櫃空間,同時,跑多實例也會減小硬件資源跑不滿的浪費。好比,百度公司的不少數據庫都是多實例,不過,通常是從庫多實例,例如某部門中使用的IBM服務器爲48核CPU,內存96GB,一臺服務器跑3~4個實例;此外,新浪網使用的也是多實例,內存48GB左右。
說明:
據調查,新浪網的數據庫單機1~4個數據庫實例的居多,其中又數1~2個的最多,由於大業務佔用的機器比較多。服務器是DELL R510的居多,CPU是E5210,48GB內存,磁盤12×300G SAS,作RAID10,此爲門戶網站的服務器配置參考,但願能給同窗們的面試帶來一些啓迪。
另外,新浪網站安裝數據庫時,通常採用編譯安裝的方式,而且會在優化以後作成rpm包,以便統一使用。
下面是MySQL官方文檔提到的單一配置文件,單一啓動程序多實例部署方案,但不推薦此方案,這裏僅做爲知識點說起,後文再也不涉及此方案的說明。my.cnf配置文件示例(MySQL手冊裏提到的方法)以下:
[mysqld_multi]
mysqld = /usr/bin/mysqld_safe
mysqladmin = /usr/bin/mysqladmin
user = mysql
[mysqld1]
socket = /var/lib/mysql/mysql.sock
port = 3306
pid-file = /var/lib/mysql/mysql.pid
datadir = /var/lib/mysql/
user = mysql
[mysqld2]
socket = /mnt/data/db1/mysql.sock
port = 3302
pid-file = /mnt/data/db1/mysql.pid
datadir = /mnt/data/db1/
user = mysql
skip-name-resolv
server-id=10
default-storage-engine=innodb
innodb_buffer_pool_size=512M
innodb_additional_mem_pool=10M
default_character_set=utf8
character_set_server=utf8
#read-only relay-log-space-limit=3G expire_logs_day=20
啓動程序的命令以下:
mysqld_multi --config-file=/data/mysql/my_multi.cnf start 1,2
該方案的缺點是耦合度過高,一個配置文件,很差管理。工做開發和運維的統一原則爲下降耦合度。
多配置文件,多啓動程序部署方案,是本文主要講解的方案,也是很是經常使用並極力推薦的多實例方案。下面來看配置示例。
[root@localhost /]# tree /data /data ├── 3306 │ ├── data #3306實例的數據目錄 │ ├── my.cnf #3306實例的配置文件 │ └── mysql #3306實例的啓動文件 └── 3307 ├── data #3307實例的數據目錄 ├── my.cnf #3307實例的配置文件 └── mysql #3307實例的啓動文件 4 directories, 4 files
提示:
這裏的配置文件my.cnf,啓動程序mysql都是獨立的文件,數據文件data目錄也是獨立的。
多實例MySQL數據庫的安裝和以前講解的單實例沒有任何區別,所以,同窗們若是有前文單實例的安裝環境,那麼能夠直接略過5.1節的內容。
1,安裝MySQL須要的依賴包和編譯軟件
(1)安裝MySQL須要的依賴包
安裝MySQL以前,最好先安裝MySQL須要的依賴包,否則後面會出現不少報錯信息,到那時還得再回來安裝MySQL的依賴包。安裝命令以下:
[root@localhost ~]# yum -y install ncurses-devel libaio-devel [root@localhost ~]# rpm -qa ncurses-devel libaio-devel ncurses-devel-5.7-4.20090207.el6.x86_64 libaio-devel-0.3.107-10.el6.x86_64
(2)安裝編譯MySQL須要的軟件
首先經過網絡得到cmake軟件,而後進行以下操做:
[root@localhost ~]# ls -lh cmake-2.8.6.tar.gz -rw-r--r-- 1 root root 5.4M 7月 19 20:43 cmake-2.8.6.tar.gz #此軟件需提早準備 [root@localhost ~]# tar xf cmake-2.8.6.tar.gz -C /usr/src/ [root@localhost ~]# cd /usr/src/cmake-2.8.6/ [root@localhost cmake-2.8.6]# ./configure [root@localhost cmake-2.8.6]# gmake && gmake install [root@localhost cmake-2.8.6]# which cmake
2,開始安裝MySQL
爲了讓同窗們學習更多的MySQL技術,接下來會以相對複雜的源代碼安裝來說解MySQL多實例的安裝。大型公司通常都會將MySQL軟件定製成rpm包,而後放到yum倉庫裏,使用yum安裝,中小企業裏的二進制和編譯安裝的區別不大。
(1)創建MySQL用戶帳號
首先以root身份登陸到Linux系統中,而後執行以下命令建立mysql用戶帳號:
[root@localhost ~]# useradd -s /sbin/nologin -M mysql [root@localhost ~]# id mysql uid=500(mysql) gid=500(mysql) 組=500(mysql)
(2)獲取MySQL軟件包
MySQL軟件包的下載地址爲:https://dev.mysql.com/downloads/mysql/
提示:
本例以MySQL編譯的方式來說解,以前已經演示過二進制方式安裝了。在生產場景中,二進制和源碼包兩種安裝方法都是能夠用的,其應用場景通常沒什麼差異。不一樣之處在於,二進制的安裝包較大,名字和源碼包也有些區別,二進制安裝過程比源碼更快。
MySQL源碼包和二進制安裝包的名稱見下圖
(3)採用編譯方式安裝MySQL
配置及編譯安裝的步驟以下:
[root@localhost ~]# tar xf mysql-5.5.22.tar.gz -C /usr/src/
[root@localhost ~]# cd /usr/src/mysql-5.5.22/
[root@localhost mysql-5.5.22]# cmake . -DCMAKE_INSTALL_PREFIX=/usr/local/mysql-5.5.22 \
> -DMYSQL_DATADIR=/usr/local/mysql-5.5.22/data \ #數據存放目錄 > -DMYSQL_UNIX_ADDR=/usr/local/mysql-5.5.22/tmp/mysql.sock \ #MySQL進程間通訊的套接字位置 > -DDEFAULT_CHARSET=utf8 \ #默認字符集爲utf8 > -DDEFAULT_COLLATION=utf8_general_ci \ #默認字符集排序規則 > -DEXTRA_CHARSETS=gbk,gb2312,utf8,ascii \ #額外的字符集支持 > -DENABLED_LOCAL_INFILE=ON \ #是否啓用加載本地數據 > -DWITH_INNOBASE_STORAGE_ENGINE=1 \ #靜態編譯innodb存儲引擎到數據庫 > -DWITH_FEDERATED_STORAGE_ENGINE=1 \ #靜態編譯FEDERATED存儲引擎到數據庫 > -DWITH_BLACKHOLE_STORAGE_ENGINE=1 \ #靜態編譯blackhole存儲引擎到數據庫 > -DWITHOUT_EXAMPLE_STORAGE_ENGINE=1 \ #不編譯EXAMPLE存儲引擎到數據庫 > -DWITHOUT_PARTITION_STORAGE_ENGINE=1 \ #不支持數據庫分區 > -DWITH_FAST_MUTEXES=1 \ > -DWITH_ZLIB=bundled \ #zlib壓縮模式 > -DENABLED_LOCAL_INFILE=1 \ #是否啓用本地的LOCAL_INFILE > -DWITH_READLINE=1 \ #使用捆綁的readline > -DWITH_EMBEDDED_SERVER=1 \ #是否要創建嵌入式服務器 > -DWITH_DEBUG=0 #禁用DEBUG(開啓影響性能) # 提示:編譯時可配置的選項不少,具體可參考官方文檔 [root@localhost mysql-5.5.22]# make && make install
下面設置不帶版本號的軟連接/usr/local/mysql,操做步驟以下:
[root@localhost mysql-5.5.22]# ln -s /usr/local/mysql-5.5.22 /usr/local/mysql [root@localhost mysql-5.5.22]# ls /usr/local/mysql bin data include lib mysql-test scripts sql-bench COPYING docs INSTALL-BINARY man README share support-files
若是上述操做未出現錯誤,查看/usr/local/mysql目錄下有內容,則MySQL5.5.22源代碼包採用cmake方式的安裝就算成功了。
在企業中,一般以/data目錄做爲MySQL多實例總的根目錄,而後規劃不一樣的數字(即MySQL實例端口號)做爲/data下面的二級目錄,不一樣的二級目錄對應的數字就做爲MySQL實例的端口號,以區別不一樣的實例,數字對應的二級目錄下包含MySQL的數據文件,配置文件及啓動文件等。
下面以配置3306,3307兩個實例爲例進行講解。建立MySQL多實例的目錄以下:
[root@localhost ~]# mkdir -p /data/{3306,3307}/data [root@localhost ~]# tree /data/ /data/ ├── 3306 #3306實例目錄 │ └── data #3306實例的數據文件目錄 ├── 3307 #3307實例目錄 └── data #3307實例的數據文件目錄 4 directories, 0 files
提示:
(1)mkdir -p /data/{3306,3307}/data至關於mkdir -p /data/3306/data;mkdir -p /data/3307/data兩條命令
(2)若是是建立多個目錄,能夠增長如3308,3309這樣的目錄名,在生產環境中,通常爲3~4個實例爲佳。
MySQL數據庫默認爲用戶提供了多個配置文件模板,用戶能夠根據服務器硬件配置的大小來選擇。
[root@localhost mysql]# ls -l support-files/my*.cnf -rw-r--r-- 1 root root 4751 7月 19 21:33 support-files/my-huge.cnf -rw-r--r-- 1 root root 19805 7月 19 21:33 support-files/my-innodb-heavy-4G.cnf -rw-r--r-- 1 root root 4725 7月 19 21:33 support-files/my-large.cnf -rw-r--r-- 1 root root 4736 7月 19 21:33 support-files/my-medium.cnf -rw-r--r-- 1 root root 2900 7月 19 21:33 support-files/my-small.cnf
注意:
這些配置文件裏的註釋很是詳細,不過是英文的。。。
上面是單實例的默認配置文件模板,若是配置多實例,和單實例會有不一樣。爲了讓MySQL多實例之間彼此獨立,要爲每個實例創建一個my.cnf配置文件和一個啓動文件MySQL,讓他們分別對應本身的數據文件目錄data。
首先,經過vim命令添加配置文件內容,命令以下:
vim /data/3306/my.cnf vim /data/3307/my.cnf
不一樣的實例須要添加的my.cnf內容會有區別,其中的配置由官方的配置模板修改而來。固然,在實際工做中,咱們是拿早已配置好的模板來進行修改的,能夠經過rz等方式上傳配置文件模板my.cnf文件到相關目錄下。
MySQL3306,3307實例配置文件以下
##實例3306配置文件my.cnf [root@localhost ~]# cat /data/3306/my.cnf [client] port = 3306 socket = /data/3306/mysql.sock [mysqld] user = mysql port = 3306 socket = /data/3306/mysql.sock basedir = /usr/local/mysql datadir = /data/3306/data open_files_limit = 1024 back_log = 600 max_connections = 800 max_connect_errors = 3000 table_open_cache = 614 external-locking = FALSE max_allowed_packet = 8M #binlog_cache_size = 1M #max_heap_table_size = 64M #read_buffer_size = 2M #read_rnd_buffer_size = 16M sort_buffer_size = 1M join_buffer_size = 1M thread_cache_size = 100 thread_concurrency = 2 query_cache_size = 2M query_cache_limit = 1M query_cache_min_res_unit = 2k #ft_min_word_len = 4 #default-storage-engine = MYISAM thread_stack = 192K transaction_isolation = READ-COMMITTED tmp_table_size = 2M max_heap_table_size = 2M #log-bin=mysql-bin #binlog_format=mixed #slow_query_log long_query_time = 1 pid-file = /data/3306/mysql.pid relay-log = /data/3306/relay-bin relay-log-info-file = /data/3306/relay-log.info binlog_cache_size = 1M max_binlog_cache_size = 1M max_binlog_size = 2M key_buffer_size = 16M read_buffer_size = 1M read_rnd_buffer_size = 1M bulk_insert_buffer_size = 1M lower_case_table_names = 1 skip-name-resolve slave-skip-errors = 1032,1062 replicate-ignore-db = mysql server-id = 1 #key_buffer_size = 32M #bulk_insert_buffer_size = 64M #myisam_sort_buffer_size = 128M #myisam_max_sort_file_size = 10G #myisam_repair_threads = 1 #myisam_recover innodb_additional_mem_pool_size = 4M innodb_buffer_pool_size = 32M innodb_data_file_path = ibdata1:128M:autoextend innodb_file_io_threads = 4 #innodb_write_io_threads = 8 #innodb_read_io_threads = 8 innodb_thread_concurrency = 8 innodb_flush_log_at_trx_commit = 2 innodb_log_buffer_size = 2M innodb_log_file_size = 4M innodb_log_files_in_group = 3 innodb_max_dirty_pages_pct = 90 innodb_lock_wait_timeout = 120 innodb_file_per_table = 0 [mysqldump] quick max_allowed_packet = 2M [mysql] no-auto-rehash #[myisamchk] #key_buffer_size = 512M #sort_buffer_size = 512M #read_buffer = 8M #write_buffer = 8M #[mysqlhotcopy] #interactive-timeout [mysqld_safe] log-error = /data/3306/mysql_yunjisuan3306.err pid-file = /data/3306/mysqld.pid
提示:實例3307的配置文件只須要將3306配置文件裏的全部3306數字替換成3307(server-id換個數字)便可。
最終完成後的多實例根/data目錄結果以下:
[root@localhost ~]# tree /data /data ├── 3306 │ ├── data │ └── my.cnf #這個就是3306實例的配置文件 └── 3307 ├── data └── my.cnf #這個就是3307實例的配置文件 4 directories, 2 files
MySQL多實例啓動文件的建立和配置文件的建立幾乎同樣,也能夠經過vim命令來添加,以下:
vim /data/3306/mysql vim /data/3307/mysql
須要添加的MySQL啓動文件內容以下。(固然,在實際工做中咱們是拿早已配置好的模板來進行修改的,能夠經過rz等方式上傳配置文件模板MySQL文件到相關目錄下)
[root@localhost ~]# cat /data/3306/mysql #!/bin/bash ############################################### #this scripts is created by Mr.chen at 2016-06-25 port=3306 mysql_user="root" mysql_pwd="" #這裏須要修改成用戶的實際密碼 CmdPath="/usr/local/mysql/bin" mysql_sock="/data/${port}/mysql.sock" #startup function function_start_mysql(){ if [ ! -e "$mysql_sock" ];then printf "Starting MySQL....\n" /bin/sh ${CmdPath}/mysqld_safe --defaults-file=/data/${port}/my.cnf 2>&1 >/dev/null & else printf "MySQL is running...\n" exit fi } #stop function function_stop_mysql(){ if [ ! -e "$mysql_sock" ];then printf "MySQL is stopped...\n" exit else printf "Stoping MySQL...\n" ${CmdPath}/mysqladmin -u ${mysql_user} -p${mysql_pwd} -S /data/${port}/mysql.sock shutdown fi } #restart function function_restart_mysql(){ printf "Restarting MySQL...\n" function_stop_mysql sleep 2 function_start_mysql } case $1 in start) function_start_mysql ;; stop) function_stop_mysql ;; restart) function_restart_mysql ;; *) printf "Usage: /data/${port}/mysql{start|stop|restart}\n" esac
3307實例的啓動文件只需修改3306啓動文件的端口便可
最終完成後的多實例根/data目錄結果以下:
[root@localhost ~]# tree /data /data ├── 3306 │ ├── data │ ├── my.cnf #3306實例的配置文件 │ └── mysql #3306實例的啓動文件 └── 3307 ├── data ├── my.cnf #3307實例的配置文件 └── mysql #3307實例的啓動文件 4 directories, 4 files
須要特別說明一下,在多實例啓動文件中,啓動MySQL不一樣實例服務,所執行的命令實質是有區別的,例如,啓動3306實例的命令以下:
mysqld_safe --defaults-file=/data/3306/my.cnf 2>&1 >/dev/null &
啓動3307實例的命令以下:mysqld_safe --defaults-file=/data/3307/my.cnf 2>&1 >/dev/null &
下面看看在多實例啓動文件中,中止MySQL不一樣實例服務的實質命令。
中止3306實例的命令以下:mysqladmin -uroot -pyunjisuan123 -S /data/3306/mysql.sock shutdown
中止3307實例的命令以下:mysqladmin -u root -pyunjisuan123 -S /data/3307/mysql.sock shutdown
1)經過下面的命令,受權mysql用戶和組管理整個多實例的根目錄/data
[root@localhost ~]# chown -R mysql.mysql /data [root@localhost ~]# find /data -name "mysql" | xargs ls -l -rw-r--r--. 1 mysql mysql 1039 Jul 20 19:33 /data/3306/mysql -rw-r--r--. 1 mysql mysql 1039 Jul 20 19:34 /data/3307/mysql
2)經過下面的命令,受權MySQL多實例全部啓動文件的mysql可執行,設置700權限最佳,注意不要用755權限,由於啓動文件裏有數據庫管理員密碼,會被讀取到。
[root@localhost ~]# find /data -name "mysql" | xargs chmod 700 [root@localhost ~]# find /data -name "mysql" | xargs ls -l -rwx------. 1 mysql mysql 1039 Jul 20 19:33 /data/3306/mysql -rwx------. 1 mysql mysql 1039 Jul 20 19:34 /data/3307/mysql
(1)配置全局路徑的意義
若是不爲MySQL的命令配置全局路徑,就沒法直接在命令行輸入mysql這樣的命令,只能用全路徑命令(/usr/local/mysql/bin/mysql),這種帶着路徑輸入命令的方式很麻煩。
(2)配置MySQL全局路徑的方法
1)確認mysql命令所在路徑,命令以下:
[root@localhost ~]# ls /usr/local/mysql/bin/mysql /usr/local/mysql/bin/mysql
2)在PATH變量前面增長/usr/local/mysql/bin路徑,並追加到/etc/profile文件中,命令以下:
[root@localhost ~]# echo 'export PATH=/usr/local/mysql/bin:$PATH' >> /etc/profile #注意,echo後邊是單引號,雙引號的話變量內容會被解析掉。 [root@localhost ~]# tail -1 /etc/profile export PATH=/usr/local/mysql/bin:$PATH [root@localhost ~]# source /etc/profile #執行source使上一行添加到/etc/profile中,內容直接生效 #以上命令的用途爲定義mysql全局路徑,實如今任意路徑執行mysql命令
提示:
更簡單的設置方法爲用下面命令作軟連接:ln -s /usr/local/mysql/bin/* /usr/local/sbin/,把mysql命令說在路徑連接到全局路徑/usr/local/sbin/的下面。
上述步驟全都配置完畢後,就能夠初始化數據庫文件了,這個步驟其實也能夠在編譯安裝MySQL以後就操做,只不過放到這裏更合理一些。
(1)初始化MySQL數據庫
初始化命令以下:
[root@localhost scripts]# ./mysql_install_db --basedir=/usr/local/mysql --datadir=/data/3306/data --user=mysql [root@localhost scripts]# ./mysql_install_db --basedir=/usr/local/mysql --datadir=/data/3307/data --user=mysql
提示:
--basedir=/usr/local/mysql爲MySQL的安裝路徑,--datadir爲不一樣的實例數據目錄
(2)初始化數據庫的原理及結果說明
初始化數據庫的實質就是建立基礎的數據庫系統的庫文件,例如:生成MySQL庫表等。
初始化數據庫後查看對應實例的數據目錄,能夠看到多了以下文件:
[root@localhost scripts]# tree /data #如下省略若干...
[root@localhost scripts]# /data/3306/mysql start Starting MySQL.... [root@localhost scripts]# /data/3307/mysql start Starting MySQL.... root@localhost scripts]# netstat -antup | grep 330 tcp 0 0 0.0.0.0:3307 0.0.0.0:* LISTEN 24743/mysqld tcp 0 0 0.0.0.0:3306 0.0.0.0:* LISTEN 24020/mysqld
從輸出中能夠看到,3306和3307實例均已正常啓動。
服務的開機自啓動很關鍵,MySQL多實例的啓動也不例外,把MySQL多實例的啓動命令加入/etc/rc.local,實現開機自啓動,命令以下:
[root@localhost ~]# echo "#mysql multi instances" >> /etc/rc.local [root@localhost ~]# echo "/data/3306/mysql start" >> /etc/rc.local [root@localhost ~]# echo "/data/3307/mysql start" >> /etc/rc.local [root@localhost ~]# tail -3 /etc/rc.local #mysql multi instances /data/3306/mysql start /data/3307/mysql start #這裏必定要確保MySQL腳本可執行~
[root@localhost ~]# mysql -S /data/3306/mysql.sock #直接敲就進來了,並且身份仍是root。可是多了-S /data/3306/mysql.sock,用於區別登錄不一樣的實例 Welcome to the MySQL monitor. Commands end with ; or \g. Your MySQL connection id is 1 Server version: 5.5.22 Source distribution Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. Oracle is a registered trademark of Oracle Corporation and/or its affiliates. Other names may be trademarks of their respective owners. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement. mysql> show databases; #查看當前全部的數據庫 +--------------------+ | Database | +--------------------+ | information_schema | | mysql | | performance_schema | | test | +--------------------+ 4 rows in set (0.00 sec) mysql> select user(); #查看當前的登錄用戶 +----------------+ | user() | +----------------+ | root@localhost | +----------------+ 1 row in set (0.00 sec)
MySQL安裝完成後,默認狀況下,MySQl管理員的帳號root是無密碼的。登錄不一樣的實例須要指定不一樣實例的mysql.sock文件路徑,這個mysql.sock是在my.cnf配置文件裏指定的。
下面是無密碼狀況下登錄數據庫的方法,關鍵點是-S參數及後面指定的/data/33306/mysql.sock,注意,不一樣實例的sock雖然名字相同,可是路徑是不一樣的,所以是不一樣的文件。
mysql -S /data/3306/mysql.sock mysql -S /sata/3307/mysql.sock
下面是重啓對應實例數據庫的命令
/data/3306/mysql stop /data/3307/mysql start
MySQL管理員的帳號root密碼默認爲空,極不安全,能夠經過mysqladmin命令爲MySQL不一樣實例的數據庫設置獨立的密碼,命令以下:
[root@localhost ~]# mysqladmin -u root -S /data/3306/mysql.sock password '123123' #爲mysql設置密碼 [root@localhost ~]# mysql -uroot -p -S /data/3306/mysql.sock #沒法直接登錄了 Enter password: Welcome to the MySQL monitor. Commands end with ; or \g. Your MySQL connection id is 2 Server version: 5.5.22 Source distribution Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. Oracle is a registered trademark of Oracle Corporation and/or its affiliates. Other names may be trademarks of their respective owners. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement. mysql> #提示:3307實例設置方法和3306實例相同,只是鏈接時的mysql,sock路徑不一樣,這已經提醒屢次,你們要注意
帶密碼登錄不一樣實例數據庫的方法:
#登錄3306實例的命令以下: mysql -uroot -p123123 -S /data/3306/mysql.sock #登錄3307實例的命令以下: mysql -uroot -p123123 -S /data/3307/mysql.sock
提示:
基礎弱的同窗,在測試時儘可能保證多實例的密碼相同,能夠減小麻煩,後面還原數據庫時會覆蓋密碼!
若要重啓多實例數據庫,也須要進行相應的以下配置。再重啓數據庫前,須要調整不一樣實例啓動文件裏對應的數據庫密碼。
[root@localhost ~]# vim /data/3306/mysql [root@localhost ~]# sed -n '7p' /data/3306/mysql #這是以前mysql多實例啓動腳本里mysql的登錄密碼變量 mysql_pwd="" [root@localhost ~]# sed -i '7 s#""#"123123"#' /data/3306/mysql [root@localhost ~]# sed -n '7p' /data/3306/mysql mysql_pwd="123123" #修改爲實際的登陸密碼 [root@localhost ~]# sed -n '7p' /data/3307/mysql mysql_pwd="" [root@localhost ~]# sed -i '7 s#""#"123123"#' /data/3307/mysql [root@localhost ~]# sed -n '7p' /data/3307/mysql mysql_pwd="123123"
多實例下正常中止數據庫的命令以下:
/data/3306/mysql stop
因爲選擇了mysqladmin shutdown的中止方式,因此中止數據庫時須要在啓動文件裏配置數據庫的密碼
/data/3306/mysql start
重點提示:
禁止使用pkill,kill -9,killall -9等命令強制殺死數據庫,這會引發數據庫沒法啓動等故障的發生。
若再3306和3307實例的基礎上,再增長一個MySQL實例,該怎麼辦?下面給出增長一個MySQL3308端口實例的命令集合:
mkdir -p /data/3308/data \cp /data/3306/my.cnf /data/3308/ \cp /data/3306/mysql /data/3308/ sed -i 's#3306#3308#g' /data/3308/my.cnf sed -i 's#server-id = 1#server-id = 8#g' /data/3308/my.cnf sed -i 's#3306#3308#g' /data/3308/mysql chown -R mysql:mysql /data/3308 chmod 700 /data/3308/mysql cd /usr/local/mysql/scripts ./mysql_install_db --datadir=/data/3308/data --basedir=/usr/local/mysql --user=mysql chown -R mysql:mysql /data/3308 egrep "server-id|log-bin" /data/3308/my.cnf /data/3308/mysql start netstat -antup | grep 3308 #提示:最好把server-id按照IP地址最後一個小數點的數字設置 #成功標誌:多了一個啓動的端口3308
若是配置之後,服務啓動後卻沒有運行起來,別忘了必定要看MySQL錯誤日誌,在/data/3308/my.cnf最下面有錯誤日誌路徑地址。
(1)多實例本地登陸MySQL
多實例本地登陸通常經過socket文件來指定具體登錄到哪一個實例,此文件的具體位置是在MySQL編譯過程或my.cnf文件中指定的。在本地登錄數據庫時,登錄程序會經過socket文件來判斷登錄的是哪一個數據庫實例。
例如:經過mysql -uroot -p '123123' -S /data/3307/mysql.sock可知,登錄的是3307這個實例。
mysql.sock文件是MySQL服務器端與本地MySQL客戶端進行通訊的UNIX套接字文件。
(2)遠程鏈接登錄MySQL多實例
遠程登錄MySQL多實例中的一個實例時,經過TCP端口(port)來指定說要登錄的MySQL實例,此端口的配置是在MySQL配置文件my.cnf中指定的。
例如:在mysql -uyunjisuan -p '123123' -h 192.168.200.101 -P 3307中,-P爲端口參數,後面接具體的實例端口,端口是一種「邏輯鏈接位置」,是客戶端程序被分派到計算機上特殊服務程序的一種方式,強調提早在192.168.200.101上對yunjisuan用戶作了受權。
MySQL數據庫的主從複製方案,與使用scp/rsync等命令進行的文件級別複製相似,都是數據的遠程傳輸,只不過MySQL的主從複製是其自帶的功能,無需藉助第三方工具,並且,MySQL的主從複製並非數據庫磁盤上的文件直接拷貝,而是經過邏輯的binlog日誌複製到要同步的服務器本地,而後由本地的線程讀取日誌裏面的SQL語句,從新應用到MySQL數據庫中。
- MySQL數據庫支持單向,雙向,鏈式級聯,環狀等不一樣業務場景的複製。在複製過程當中,一臺服務器充當主服務器(Master),接收來自用戶的內容更新,而一個或多個其餘的服務器充當從服務器(Slave),接收來自主服務器binlog文件的日誌內容,解析出SQL,從新更新到從服務器,使得主從服務器數據達到一致。
- 若是設置了鏈式級聯複製,那麼,從服務器(Slave)自己除了充當從服務器外,也會同時充當其下面從服務器的主服務器。鏈式級聯複製相似A-->B-->C的複製形式。
下圖爲單向主從複製架構邏輯圖,此架構只能在Master端進行數據寫入
下圖爲雙向主主複製邏輯架構圖,此架構能夠在Master1端或Master2端進行數據寫入,或者兩端同時寫入數據(須要特殊設置)
下圖爲線性級聯單向雙主複製邏輯架構圖,此架構只能在Master1端進行數據寫入,工做場景中,Master1和master2做爲主主互備,Slave1做爲從庫,中間的Master2須要作特殊的設置。
下圖爲環狀級聯單向多主同步邏輯架構圖,任意一個點均可以寫入數據,此架構比較複雜,屬於極端環境下的「成品」,通常場景慎用
在當前的生產工做中,MySQL主從複製都是異步的複製方式,既不是嚴格實時的數據同步,可是正常狀況下給用戶的體驗是真實的。
MySQL主從複製集羣功能使得MySQL數據庫支持大規模高併發讀寫成爲可能,同時有效地保護了物理服務器宕機場景的數據備份。
應用場景1:從服務器做爲主服務器的實時數據備份
- 主從服務器架構的設置能夠大大增強MySQL數據庫架構的健壯性。例如:當主服務器出現問題時,咱們能夠人工或設置自動切換到從服務器繼續提供服務,此時從服務器的數據與宕機時的主數據庫幾乎是一致的。
- 這相似NFS存儲數據經過inotify + rsync同步到備份的NFS服務器,只不過MySQL的複製方案是其自帶的工具。
- 利用MySQL的複製功能進行數據備份時,在硬件故障,軟件故障的場景下,該數據備份是有效的,但對於人爲地執行drop,delete等語句刪除數據的狀況,從庫的備份功能就沒用了,由於從服務器也會執行刪除的語句。
應用場景2:主從服務器實現讀寫分離,從服務器實現負載均衡
- 主從服務器架構可經過程序(PHP,java等)或代理軟件(mysql-proxy,Amoeba)實現對用戶(客戶端)的請求讀寫分離,即讓從服務器僅僅處理用戶的select查詢請求,下降用戶查詢響應時間,以及同時讀寫在主服務器上帶來的訪問壓力。對於更新的數據(例如:update,insert,delete語句),則仍然交給主服務器處理,確保主服務器和從服務器保持實時同步。
- 百度,淘寶,新浪等絕大多數的網站都是用戶瀏覽頁面多於用戶發佈內容,所以經過在從服務器上接收只讀請求,就能夠很好地減輕主庫的讀壓力,且從服務器能夠很容易地擴展爲多臺,使用LVS作負載均衡效果就很是棒了,這就是傳說中的數據庫讀寫分離架構。邏輯架構圖以下所示:
應用場景3:把多個從服務器根據業務重要性進行拆分訪問
能夠把幾個不一樣的從服務器,根據公司的業務進行拆分。例如:有爲外部用戶提供查詢服務的從服務器,有內部DBA用來數據備份的從服務器,還有爲公司內部人員提供訪問的後臺,腳本,日誌分析及供開發人員查詢使用的從服務器。這樣的拆分除了減輕主服務器的壓力外,還可使數據庫對外部用戶瀏覽,內部用戶業務處理及DBA人員的備份等互不影響。
(1)經過程序實現讀寫分離(性能和效率最佳,推薦)
PHP和Java程序均可以經過設置多個鏈接文件輕鬆地實現對數據庫的讀寫分離,即當語句關鍵字爲select時,就去鏈接讀庫的鏈接文件,若爲update,insert,delete時,則鏈接寫庫的鏈接文件。
經過程序實現讀寫分離的缺點就是須要開發人員對程序進行改造,使其對下層不透明,但這種方式更容易開發和實現,適合互聯網業務場景。
根據業務重要性拆分從庫方案
(2)經過開源的軟件實現讀寫分離
MySQL-proxy,Amoeba等代理軟件也能夠實現讀寫分離功能,這些軟件的穩定性和功能通常,不建議生產使用。絕大多數公司經常使用的仍是在應用端發程序實現讀寫分離。
(3)大型門戶獨立開發DAL層綜合軟件
百度,阿里等大型門戶都有開發牛人,會花大力氣開發適合本身業務的讀寫分離,負載均衡,監控報警,自動擴容,自動收縮等一系列功能的DAL層軟件。
- MySQL的主從複製是一個異步的複製過程(雖然通常狀況下感受是實時的),數據將從一個MySQL數據庫(咱們稱之爲Master)複製到另外一個MySQL數據庫(咱們稱之爲Slave),在Master與Slave之間實現整個主從複製的過程是由三個線程參與完成的。其中有兩個線程(SQL線程和I/O線程)在Slave端,另一個線程(I/O線程)在Master端。
- 要實現MySQL的主從複製,首先必須打開Master端的binlog記錄功能,不然就沒法實現。由於整個複製過程實際上就是Slave從Master端獲取binlog日誌,而後再在Slave上以相同順序執行獲取的binlog日誌中所記錄的各類SQL操做。
- 要打開MySQL的binlog記錄功能,可經過在MySQL的配置文件my.cnf中的mysqld模塊([mysqld]標識後的參數部分)增長「log-bin」參數選項來實現,具體信息以下。
[mysqld] log-bin=/data/3306/mysql-bin
提示:
有些同窗把log-bin放在了配置文件結尾,而不是[mysqld]標識後,從而致使配置複製不成功。
下面簡單描述MySQL Replication的複製原理過程
1)在Slave服務器上執行start slave命令開啓主從複製開關,開始進行主從複製
2)此時,Slave服務器的I/O線程會經過在Master上已經受權的複製用戶權限請求鏈接Master服務器,並請求從指定binlog日誌文件的指定位置(日誌文件名和位置就是在配置主從複製服務時執行change master命令指定的)以後開始發送binlog日誌內容。
3)Master服務器接收到來自Slave服務器的I/O線程的請求後,其上負責複製的I/O線程會根據Slave服務器的I/O線程請求的信息分批讀取指定binlog日誌文件指定位置以後的binlog日誌信息,而後返回給Slave端的I/O線程。返回的信息中除了binlog日誌內容外,還有在Master服務器端記錄的新的binlog文件名稱,以及在新的binlog中的下一個指定更新位置。
4)當Slave服務器的I/O線程獲取到Master服務器上I/O線程發送的日誌內容,日誌文件及位置點後,會將binlog日誌內容依次寫到Slave端自身的Relay Log(即中繼日誌)文件(MySQL-relay-bin.xxxx)的最末端,並將新的binlog文件名和位置記錄到master-info文件中,以便下一次讀取Master端新binlog日誌時可以告訴Master服務器重新binlog日誌的指定文件及位置開始請求新的binlog日誌內容。
5)Slave服務器端的SQL線程會實時檢測本地Relay Log中I/O線程新增長的日誌內容,而後及時地把Relay Log文件中的內容解析成SQL語句,並在自身Slave服務器上按解析SQL語句的位置順序執行應用這些SQL語句,並在relay-log.info中記錄當前應用中繼日誌的文件名及位置點。
通過了上面的過程,就能夠確保在Master端和Slave端執行了一樣的SQL語句。當複製狀態正常時,Master端和Slave端的數據是徹底同樣的。固然,MySQL的複製機制也有一些特殊狀況,具體請參考官方的說明,大多數狀況下,同窗們不用擔憂。
MySQL Replication的複製原理邏輯圖
特別說明:
當企業面試MySQL主從複製原理時,無論是面試仍是筆試,都要儘可能畫圖表達,而不是口頭講或文字描述,面試時能夠找黑板或拿出紙來給面試官詳細講解。
下面針對MySQL主從複製原理的重點進行小結:
MySQL主從複製實踐對環境的要求比較簡單,能夠是單機單數據庫多實例的環境,也能夠是兩臺服務器,每一個機器一個獨立數據庫的環境。本文以單機數據庫多實例的環境爲例講解。實例端口信息查看以下:
[root@localhost ~]# ss -antup | grep 330 tcp LISTEN 0 128 *:3307 *:* users:(("mysqld",3910,11)) tcp LISTEN 0 128 *:3306 *:* users:(("mysqld",2450,11))
提示:
這裏把3306實例做爲主庫,3307實例做爲從庫,若是根據前面的內容配置了MySQL多實例環境,直接開啓多實例環境使用便可。
這裏的主從複製技術是針對前面的內容以單機數據庫多實例環境來說的。通常狀況下,小企業在作常規的主從複製時,主從服務器多數在不一樣的機器上,而且監聽的端口均爲默認的3306.雖然不在同一個機器上,可是步驟和過程倒是同樣的。
同窗們在掌握了但數據庫多實例的同步方法後,能夠本身適當擴展,完成異機相同端口之間的主從複製。
根據以前介紹的MySQL主從複製原理咱們知道,要實現主從複製,關鍵是要開啓binlog日誌功能,因此,首先來打開主庫的binlog日誌參數。
1)修改主庫的配置文件。執行vi /data/3306/my.cnf,編輯多實例3006的my.cnf配置文件,按以下內容修改兩個參數:
[mysqld] server-id = 1 #用於同步的每臺機器或實例server-id都不能相同 log-bin = /data/3306/mysql-bin #binlog日誌的位置
提示:
上面的兩個參數要放在my.cnf中的[mysqld]模塊下,不然會出錯。
不一樣實例間server-id的值不能夠重複
要先在my.cnf配置文件中查找相關參數,並按要求修改。若發現不存在,再添加參數,切記,參數不能重複。
修改my.cnf配置後,須要重啓動數據庫,命令爲:/data/3306/mysql restart ,注意要確認真正重啓了。
2)檢查配置參數後的結果,以下:
root@localhost ~]# egrep "server-id|log-bin" /data/3306/my.cnf server-id = 1 log-bin=/data/3306/mysql-bin #log-bin後面也能夠不帶等號內容,MySQL會使用默認日誌
3)重啓主庫MySQL服務,命令以下:
[root@localhost ~]# /data/3306/mysql restart Restarting MySQL... Stoping MySQL... Starting MySQL....
4)登錄數據庫,檢查參數的更改狀況,以下:
[root@localhost ~]# mysql -uroot -p123123 -S /data/3306/mysql.sock #登錄3306實例 Welcome to the MySQL monitor. Commands end with ; or \g. Your MySQL connection id is 1 Server version: 5.5.22-log Source distribution Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. Oracle is a registered trademark of Oracle Corporation and/or its affiliates. Other names may be trademarks of their respective owners. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement. mysql> show variables like 'server_id'; #查看MySQL的系統變量(like相似於grep過濾) +---------------+-------+ | Variable_name | Value | +---------------+-------+ | server_id | 1 | #配置的server_id爲1 +---------------+-------+ 1 row in set (0.00 sec) mysql> show variables like 'log_bin'; +---------------+-------+ | Variable_name | Value | +---------------+-------+ | log_bin | ON | #binlog功能已開啓 +---------------+-------+ 1 row in set (0.00 sec) mysql> #這樣,binlog功能就開啓了。
根據主從複製的原理,從庫要想和主庫同步,必須有一個能夠鏈接主庫的帳號,而且這個帳號是主庫上建立的,權限是容許主庫的從庫鏈接並同步數據。
1)登錄MySQL3306實例主數據庫,命令以下:
[root@localhost ~]# mysql -uroot -p123123 -S /data/3306/mysql.sock
2)創建用於從庫複製的帳號yunjisuan,命令以下:
mysql> grant replication slave on *.* to 'yunjisuan'@'192.168.0.%' identified by 'yunjisuan123'; Query OK, 0 rows affected (0.00 sec) #語句說明: 1)replication slave爲mysql同步的必須權限,此處不要受權all權限 2)*.* 表示全部庫全部表,也能夠指定具體的庫和表進行復制。例如yunjisuan.test中,yunjisuan爲庫名,test爲表名 3)'yunjisuan'@'192.168.0.%' yunjisuan爲同步帳號。192.168.0.%爲受權主機網段,使用了%表示容許整個192.168.0.0網段能夠用yunjisuan這個用戶訪問數據庫 4)identified by 'yunjisuan123'; yunjisuan123爲密碼,實際環境下設置的複雜些爲好
建立完帳號並受權後,須要刷新權限,使受權的權限生效
mysql> flush privileges; #刷新權限 Query OK, 0 rows affected (0.00 sec)
3)檢查主庫建立的yunjisuan複製帳號命令及結果以下:
mysql> select user,host from mysql.user;
+-----------+-------------+
| user | host |
+-----------+-------------+
| root | 127.0.0.1 | | yunjisuan | 192.168.0.% | #出現這行表示複製帳號已經配置好了 | root | ::1 | | | localhost | | root | localhost | +-----------+-------------+ 5 rows in set (0.00 sec) #說明: MySQL裏的受權用戶是以數據表格的形式存儲在mysql這個庫的user表裏。
mysql> select user,host from mysql.user where user='yunjisuan'; #where是SQL查詢語句的條件 +-----------+-------------+ | user | host | +-----------+-------------+ | yunjisuan | 192.168.0.% | +-----------+-------------+ 1 row in set (0.00 sec) mysql> show grants for yunjisuan@'192.168.0.%'; #查看帳號的受權狀況 +--------------------------------------------------------------------------------------------------------------------------------+ | Grants for yunjisuan@192.168.0.% | +--------------------------------------------------------------------------------------------------------------------------------+ | GRANT REPLICATION SLAVE ON *.* TO 'yunjisuan'@'192.168.0.%' IDENTIFIED BY PASSWORD '*A2CC7FA422EF5A7CB098FEA7732C1F78CDC32F67' | #結果顯示受權正確 +--------------------------------------------------------------------------------------------------------------------------------+ 1 row in set (0.00 sec)
1)對主數據庫鎖表只讀(當前窗口不要關掉)的命令以下:
mysql> flush table with read lock; Query OK, 0 rows affected (0.00 sec)
提示:
在引擎不一樣的狀況下,這個鎖表命令的時間會受下面參數的控制。鎖表時,若是超過設置時間不操做會自動解鎖。
默認狀況下自動解鎖的時長參數值以下:
mysql> show variables like '%timeout%'; +----------------------------+----------+ | Variable_name | Value | +----------------------------+----------+ | connect_timeout | 10 | | delayed_insert_timeout | 300 | | innodb_lock_wait_timeout | 120 | | innodb_rollback_on_timeout | OFF | | interactive_timeout | 28800 | #自動解鎖時間受本參數影響 | lock_wait_timeout | 31536000 | | net_read_timeout | 30 | | net_write_timeout | 60 | | slave_net_timeout | 3600 | | wait_timeout | 28800 | #自動解鎖時間受本參數影響 +----------------------------+----------+ 10 rows in set (0.00 sec) #提示:有關這兩個參數,請同窗們自行測試
2)鎖表後查看主庫狀態。可經過當前binlog日誌文件名和二進制binlog日誌偏移量來查看,結果以下:
注意,show master status;命令顯示的信息要記錄在案,後面的從庫導入全備後,繼續和主庫複製時就是要從這個位置開始。
mysql> show master status;
+------------------+----------+--------------+------------------+
| File | Position | Binlog_Do_DB | Binlog_Ignore_DB | +------------------+----------+--------------+------------------+ | mysql-bin.000001 | 345 | | | +------------------+----------+--------------+------------------+ 1 row in set (0.00 sec)
或者新開一個命令行窗口,用以下命令查看鎖表後的主庫binlog位置點信息:
[root@localhost ~]# mysql -uroot -p123123 -S /data/3306/mysql.sock -e "show master status" +------------------+----------+--------------+------------------+ | File | Position | Binlog_Do_DB | Binlog_Ignore_DB | +------------------+----------+--------------+------------------+ | mysql-bin.000001 | 533 | | | +------------------+----------+--------------+------------------+
3)鎖表後,必定要單開一個新的SSH窗口,導出數據庫的全部數據,若是數據量很大(50GB以上),而且容許停機,能夠停庫直接打包數據文件進行遷移,那樣更快。
[root@localhost ~]# mkdir -p /server/backup [root@localhost ~]# mysqldump -uroot -p123123 -S /data/3306/mysql.sock --events -A -B | gzip >/server/backup/mysql_bak.$(date +%F).sql.gz #注意:-A表示備份全部庫;-B表示增長use DB和 drop 等(導庫時會直接覆蓋原有的) [root@localhost ~]# ll /server/backup/mysql_bak.2017-07-21.sql.gz -rw-r--r--. 1 root root 137344 Jul 21 10:17 /server/backup/mysql_bak.2017-07-21.sql.gz #爲了確保導出數據期間,數據庫沒有數據插入,導庫完畢能夠再次檢查主庫狀態信息,結果以下: [root@localhost ~]# mysql -uroot -p123123 -S /data/3306/mysql.sock -e "show master status" +------------------+----------+--------------+------------------+ | File | Position | Binlog_Do_DB | Binlog_Ignore_DB | +------------------+----------+--------------+------------------+ | mysql-bin.000001 | 533 | | | +------------------+----------+--------------+------------------+ 提示:若無特殊狀況,binlog文件及位置點和鎖表後導出數據前是一致的,即沒有變化。 #導出數據完畢後,解鎖主庫,恢復可寫,命令以下.由於主庫還要對外提供服務,不能一直鎖定不讓用戶訪問。 mysql> unlock tables; Query OK, 0 rows affected (0.00 sec) #可能會有同窗由於鎖表後的binlog位置問題犯迷糊,實際上作從庫前,不管主庫更新了多少數據,最後從庫均可以從上面show master status的位置很快遇上主庫的進度。
下面主要講解單數據庫多實例的主從配置,也就是說,mysqldump備份的3306實例的數據和要恢復的3307實例在一臺機器上,所以無需異地複製拷貝。想查看主庫導出的數據,以下:
[root@localhost ~]# ll /server/backup/mysql_bak.2017-07-21.sql.gz -rw-r--r--. 1 root root 137344 Jul 21 10:17 /server/backup/mysql_bak.2017-07-21.sql.gz
- 數據庫的server-id通常在一套主從複製體系內是惟一的,這裏從庫的server-id要和主庫及其餘從庫的不一樣,而且要註釋掉從庫的binlog參數配置,若是從庫不作級聯複製,而且不做爲備份用,就不要開啓binlog,開啓了反而會增長從庫磁盤I/O等的壓力。
- 可是,有如下兩種狀況須要打開從庫的binlog記錄功能,記錄數據庫更新的SQL語句:
- 級聯同步A-->B-->C中間的B時,就要開啓binlog記錄功能。
- 在從庫作數據庫備份,數據庫備份必需要有全備和binlog日誌,纔是完整的備份。
(1)修改配置文件,配置從庫1的相關參數
執行vi /data/3307/my.cnf,編輯my.cnf配置文件,按以下內容修改兩個參數:
[mysqld] server-id = 3 #調整等號後的數值,和任何一個數據庫實例都不一樣
提示:
上面兩參數要放在my.cnf中的[mysqld]模塊下,不然會出錯。
server-id的值不能和任何MySQL實例重複。
要先在文件中查找相關參數按要求修改。若發現不存在,再添加參數,切記,參數不能重複。
修改my.cnf配置後須要重啓數據庫,命令爲:/data/3307/mysql restart,注意要確認真正重啓了。
(2)檢查配置參數後的結果
命令以下:
[root@localhost ~]# egrep "server-id|log-bin" /data/3307/my.cnf server-id = 3
(3)重啓3307的從數據庫
命令以下:
[root@localhost ~]# /data/3307/mysql restart Restarting MySQL... Stoping MySQL... Starting MySQL.... [root@localhost ~]# ss -antup | grep 3307 tcp LISTEN 0 128 *:3307 *:* users:(("mysqld",5659,11))
(4)登錄數據庫檢查參數的改變狀況
命令以下:
[root@localhost ~]# mysql -uroot -p123123 -S /data/3307/mysql.sock Welcome to the MySQL monitor. Commands end with ; or \g. Your MySQL connection id is 1 Server version: 5.5.22 Source distribution Copyright (c) 2000, 2011, Oracle and/or its affiliates. All rights reserved. Oracle is a registered trademark of Oracle Corporation and/or its affiliates. Other names may be trademarks of their respective owners. Type 'help;' or '\h' for help. Type '\c' to clear the current input statement. mysql> show variables like 'log_bin'; +---------------+-------+ | Variable_name | Value | +---------------+-------+ | log_bin | OFF | +---------------+-------+ 1 row in set (0.00 sec) mysql> show variables like 'server_id'; +---------------+-------+ | Variable_name | Value | +---------------+-------+ | server_id | 3 | +---------------+-------+ 1 row in set (0.00 sec) mysql>
操做命令以下:
[root@localhost ~]# cd /server/backup/ [root@localhost backup]# ls -l total 136 -rw-r--r--. 1 root root 137344 Jul 21 10:17 mysql_bak.2017-07-21.sql.gz [root@localhost backup]# gzip -d mysql_bak.2017-07-21.sql.gz [root@localhost backup]# ll total 496 -rw-r--r--. 1 root root 506730 Jul 21 10:17 mysql_bak.2017-07-21.sql [root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock <mysql_bak.2017-07-21.sql #這是把數據還原到3307實例的命令 #提示: 若是備份時使用了-A參數,則在還原數據到3307實例時,登錄3307實例的密碼也會和3306主庫的一致,由於3307實例的受權表MySQL也被覆蓋了。
(1)MySQL從庫鏈接主庫的配置信息以下:
CHANGE MASTER TO
MASTER_HOST='192.168.0.200', #這裏是主庫的IP MASTER_PORT=3306, #這裏是主庫的端口,從庫端口能夠和主庫不一樣 MASTER_USER='yunjisuan', #這裏是主庫上創建的用於複製的用戶yunjisuan MASTER_PASSWORD='yunjisuan123', #這裏是yunjisuan用戶的密碼 MASTER_LOG_FILE='mysql-bin.000001', #這裏是show master status時查看到的二進制日誌文件名稱,注意不能多空格 MASTER_LOG_POS=533; #這裏是show master status時查看到的二進制日誌偏移量,注意不能多空格 #提示:字符串用單引號括起來,數值不用引號,注意內容先後不能有空格。
(2)登錄數據庫後,去掉上述語句中的註釋,執行以下:
mysql> CHANGE MASTER TO MASTER_HOST='192.168.0.200',MASTER_PORT=3306,MASTER_USER='yunjisuan',MASTER_PASSWORD='yunjisuan123',MASTER_LOG_FILE='mysql-bin.000001',MASTER_LOG_POS=533; #提示:這個步驟的參數必定不能錯,不然,數據庫複製配置會失敗
上述操做的原理其實是把用戶密碼等信息寫入從庫新的master.info文件中
[root@localhost backup]# ll /data/3307/data/master.info -rw-rw----. 1 mysql mysql 90 Jul 21 11:31 /data/3307/data/master.info [root@localhost backup]# cat /data/3307/data/master.info 18 mysql-bin.000001 #這裏是show master status時查看的二進制日誌文件名稱 533 #這裏是show master status時查看的二進制日誌偏移量 192.168.0.200 #這裏是主庫的IP yunjisuan #這裏是主庫上創建的用於複製的用戶yunjisuan yunjisuan123 #這裏是yunjisuan用戶的密碼 3306 #這裏是主庫的端口 60 0...如下省略若干...
(1)啓動從庫主從複製開關,並查看複製狀態
相關語句以下:
[root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock -e "start slave" [root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock -e "show slave status\G" *************************** 1. row *************************** Slave_IO_State: Waiting for master to send event Master_Host: 192.168.0.200 Master_User: yunjisuan Master_Port: 3306 Connect_Retry: 60 Master_Log_File: mysql-bin.000001 Read_Master_Log_Pos: 533 Relay_Log_File: relay-bin.000002 Relay_Log_Pos: 253 Relay_Master_Log_File: mysql-bin.000001 Slave_IO_Running: Yes Slave_SQL_Running: Yes Replicate_Do_DB: Replicate_Ignore_DB: mysql Replicate_Do_Table: Replicate_Ignore_Table: Replicate_Wild_Do_Table: Replicate_Wild_Ignore_Table: Last_Errno: 0 Last_Error: Skip_Counter: 0 Exec_Master_Log_Pos: 533 Relay_Log_Space: 403 Until_Condition: None Until_Log_File: Until_Log_Pos: 0 Master_SSL_Allowed: No Master_SSL_CA_File: Master_SSL_CA_Path: Master_SSL_Cert: Master_SSL_Cipher: Master_SSL_Key: Seconds_Behind_Master: 0 Master_SSL_Verify_Server_Cert: No Last_IO_Errno: 0 Last_IO_Error: Last_SQL_Errno: 0 Last_SQL_Error: Replicate_Ignore_Server_Ids: Master_Server_Id: 1
主從同步是否成功,最關鍵的爲下面的3項狀態參數:
[root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock -e "show slave status\G" | egrep "IO_Running|SQL_Running|Seconds_Behind_Master" Slave_IO_Running: Yes Slave_SQL_Running: Yes Seconds_Behind_Master: 0
(2)測試主從複製結果
在主庫上寫入數據,而後觀察從庫的數據情況。
[root@localhost backup]# mysql -uroot -p123123 -S /data/3306/mysql.sock -e "create database benet" [root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock -e "show databases" +--------------------+ | Database | +--------------------+ | information_schema | | benet | | mysql | | performance_schema | | test | +--------------------+ [root@localhost backup]# mysql -uroot -p123123 -S /data/3306/mysql.sock -e "create database yunjisuan" [root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock -e "show databases" +--------------------+ | Database | +--------------------+ | information_schema | | benet | | mysql | | performance_schema | | test | | yunjisuan | +--------------------+ [root@localhost backup]# mysql -uroot -p123123 -S /data/3306/mysql.sock -e "drop database yunjisuan" [root@localhost backup]# mysql -uroot -p123123 -S /data/3307/mysql.sock -e "show databases" +--------------------+ | Database | +--------------------+ | information_schema | | benet | | mysql | | performance_schema | | test | +--------------------+ #根據測試能夠判斷,主從庫是同步的。
MySQL主從複製配置完整步驟以下:
(1)登錄主數據庫查看MySQL線程的同步狀態
mysql> show processlist\G *************************** 1. row *************************** Id: 1 User: root Host: localhost db: NULL Command: Query Time: 0 State: NULL Info: show processlist *************************** 2. row *************************** Id: 5 User: yunjisuan Host: 192.168.0.200:42008 db: NULL Command: Binlog Dump Time: 267 State: Master has sent all binlog to slave; waiting for binlog to be updated Info: NULL 2 rows in set (0.00 sec) #提示:上述狀態的意思是線程已經從binlog日誌讀取全部更新,並已經發送到了從數據庫服務器。線程目前爲空閒狀態,等待由主服務器上二進制日誌中的新事件更新。
下圖中列出了主服務器binlog Dump線程中State列的最多見狀態。若是你沒有在主服務器上看見任何binlog Dump線程,則說明覆制沒有運行,二進制binlog日誌由各類事件組成,事件一般會爲更新添加信息。
(2)登錄從數據庫查看MySQL線程工做狀態
從庫有兩個線程,即I/O和SQL線程。從庫I/O線程的狀態以下:
mysql> show processlist\G *************************** 1. row *************************** Id: 3 User: root Host: localhost db: NULL Command: Query Time: 0 State: NULL Info: show processlist *************************** 2. row *************************** Id: 8 User: system user Host: db: NULL Command: Connect Time: 659 State: Waiting for master to send event Info: NULL *************************** 3. row *************************** Id: 9 User: system user Host: db: NULL Command: Connect Time: 511 State: Slave has read all relay log; waiting for the slave I/O thread to update it Info: NULL 3 rows in set (0.00 sec)
下圖列出了從服務器的I/O線程的State列的最多見的狀態。該狀態也出如今Slave_IO_State列,由SHOW SLAVE STATUS顯示。
下圖列出了從服務器的SQL線程的State列的最多見狀態
- 經過MySQL線程同步狀態能夠看到同步是否正常進行,故障的位置是什麼,另外還可查看數據庫同步是否完成,可用於主庫宕機切換數據庫或人工數據庫主從切換遷移等。
- 例如:主庫宕機,要選擇最快的從庫將其提高爲主庫,就須要查看主從庫的線程狀態,若是主從複製在正常狀況下進行角色切換,也須要查看主從庫的線程狀態,根據複製狀態肯定更新是否完成。
模擬重現故障的能力是運維人員最重要的能力。下面就來次模擬操做。先在從庫建立一個庫,而後去主庫建立同名的庫來模擬數據衝突,命令以下:
mysql> show slave status\G *************************** 1. row *************************** Slave_IO_State: Waiting for master to send event Master_Host: 192.168.0.200 Master_User: yunjisuan Master_Port: 3306 Connect_Retry: 60 Master_Log_File: mysql-bin.000003 Read_Master_Log_Pos: 544 Relay_Log_File: relay-bin.000010 Relay_Log_Pos: 336 Relay_Master_Log_File: mysql-bin.000003 Slave_IO_Running: Yes Slave_SQL_Running: No Replicate_Do_DB: Replicate_Ignore_DB: mysql Replicate_Do_Table: Replicate_Ignore_Table: Replicate_Wild_Do_Table: Replicate_Wild_Ignore_Table: Last_Errno: 1007 Last_Error: Error 'Can't create database 'yunjisuan'; database exists' on query. Default database: 'yunjisuan'. Query: 'create database yunjisuan' Skip_Counter: 0 Exec_Master_Log_Pos: 451 Relay_Log_Space: 810 Until_Condition: None Until_Log_File: Until_Log_Pos: 0 Master_SSL_Allowed: No Master_SSL_CA_File: Master_SSL_CA_Path: Master_SSL_Cert: Master_SSL_Cipher: Master_SSL_Key: Seconds_Behind_Master: NULL Master_SSL_Verify_Server_Cert: No Last_IO_Errno: 0 Last_IO_Error: Last_SQL_Errno: 1007 Last_SQL_Error: Error 'Can't create database 'yunjisuan'; database exists' on query. Default database: 'yunjisuan'. Query: 'create database yunjisuan' Replicate_Ignore_Server_Ids: Master_Server_Id: 1 1 row in set (0.00 sec)
對於該衝突,解決辦法以下
辦法一:關閉從同步,調動sql_slave指針
mysql> stop slave; Query OK, 0 rows affected, 1 warning (0.00 sec) mysql> set global sql_slave_skip_counter=1; #將sql線程同步指針向下移動一個,若是屢次不一樣步,能夠重複操做 Query OK, 0 rows affected (0.00 sec) mysql> start slave; Query OK, 0 rows affected (0.00 sec)
對於普通的互聯網業務,上述的移動指針的操做帶來的問題不是很大。固然,要在確認不影響公司業務的前提下。
如果在企業場景下,對當前業務來講,解決主從同步比主從不一致更重要,若是主從數據一致也是很重要的,那就再找個時間恢復這個從庫。
是主從數據不一致更重要,仍是保持主從同步持續狀態更重要,要根據業務選擇。這樣Slave就會與Master同步了,主要關鍵點以下:
Slave_IO_Running: Yes Slave_SQL_Running: Yes Seconds_Behind_Master: 0 #0表示已經同步狀態 #提示:set global sql_slave_skip_counter=n; #n取值>0,忽略執行N個更新。
辦法二:根據能夠忽略的錯誤號事先在配置文件中配置,跳過指定的不影響業務數據的錯誤,例如:
[root@localhost ~]# grep slave-skip /data/3306/my.cnf slave-skip-errors = 1032,1062 #提示:相似因爲入庫重複致使的失敗能夠忽略,其餘狀況是否是能夠忽略須要根據不一樣公司的具體業務來評估。
其餘可能引發複製故障的緣由:
從庫須要記錄binlog的應用場景:當前的從庫還要做爲其餘從庫的主庫,例如級聯複製或雙主互爲主從場景的狀況下。下面介紹從庫記錄binlog日誌的方法。
在從庫的my.cnf中加入以下參數,而後重啓服務生效便可。
log-slave-updates #必需要有這個參數 log-bin = /data/3307/mysql-bin expire_logs_days = 7 #至關於find /data/3307/ -type f -name "mysql-bin.000*" -mtime +7 | xargs rm -f
步驟以下:
1)選擇一個不對外提供服務的從庫,這樣能夠確保和主庫更新最接近,專門用於作數據備份。
2)開啓從庫的binlog功能
備份時能夠選擇只中止SQL線程,中止應用SQL語句到數據庫,I/O線程保留工做狀態,執行命令爲stop slave sql_thread;,備份方式能夠採起mysqldump邏輯備份或直接物理備份,例如:使用cp,tar(針對/data目錄)工具或xtrabackup(第三方的物理備份軟件)進行備份,則邏輯備份和物理備份的選擇,通常是根據總的備份數據量的多少進行選擇的,數據量低於30G,建議選擇mysqldump邏輯備份方法,安全穩定,最後把全備和binlog數據發送到備份服務器上留存。
問題一:主庫的從庫太多,致使複製延遲
從庫數量以3~5個爲宜,要複製的從節點數量過多,會致使複製延遲。
問題二:從庫硬件比主庫差,致使複製延遲。
查看Master和Slave的系統配置,可能會由於機器配置不當,包括磁盤I/O,CPU,內存等各方面因素形成複製的延遲。這通常發生在高併發大數據量寫入場景中。
問題三:慢SQL語句太多
假如一條SQL語句執行時間是20秒,那麼從執行完畢到從庫上能查到數據至少須要20秒,這樣就延遲20秒了。
通常要把SQL語句的優化做爲常規工做,不斷的進行監控和優化,若是單個SQL的寫入時間長,能夠修改後分屢次寫入。經過查看慢查詢日誌或show full processlist命令,找出執行時間長的查詢語句或大的事務。
問題四:主從複製的設計問題
例如,主從複製單線程,若是主庫寫併發太大,來不及傳送到從庫,就會致使延遲。
更高版本的MySQL能夠支持多線程複製,門戶網站則會本身開發多線程同步功能。
問題五:主從庫之間的網絡延遲
主從庫的網卡,網線,鏈接的交換機等網絡設備均可能成爲複製的瓶頸,致使複製延遲,另外,跨公網主從複製很容易致使主從複製延遲。
問題六:主庫讀寫壓力大,致使複製延遲。
主庫硬件要搞好一點,架構的前端要加buffer及緩存層。
read-only參數選項可讓從服務器只容許來自從服務器線程或具備SUPER權限的數據庫用戶進行更新,確保從服務器不接受來自用戶端的非法用戶更新。
read-only參數容許數據庫更新的條件爲:
read-only參數的配置方法以下:
方法一:直接帶 --read-only參數啓動或重啓數據庫,
使用killall mysqld
或mysqladmin -uroot -p123123 -S /data/3307/mysql.sock shutdown
mysqld_safe --defaults-file=/data/3307/my.cnf --read-only &
方法二:在my.cnf裏[mysqld]模塊下加read-only參數重啓數據庫,配置以下:
[mysqld]
read-only