在MySQL中間件出現(xiàn)之前,對于MySQL主從集群,如果要實現(xiàn)其讀寫分離,一般是在程序端實現(xiàn),這樣就帶來一個問題,即數(shù)據(jù)庫和程序的耦合度太高,如果我數(shù)據(jù)庫的地址發(fā)生改變了,那么我程序端也要進行相應(yīng)的修改,如果數(shù)據(jù)庫不小心掛掉了,則同時也意味著程序的不可用,而這對很多應(yīng)用來說,并不能接受。
江南網(wǎng)站制作公司哪家好,找成都創(chuàng)新互聯(lián)!從網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、APP開發(fā)、響應(yīng)式網(wǎng)站開發(fā)等網(wǎng)站項目制作,到程序開發(fā),運營維護。成都創(chuàng)新互聯(lián)從2013年成立到現(xiàn)在10年的時間,我們擁有了豐富的建站經(jīng)驗和運維經(jīng)驗,來保證我們的工作的順利進行。專注于網(wǎng)站建設(shè)就選成都創(chuàng)新互聯(lián)。
引入MySQL中間件能很好的對程序端和數(shù)據(jù)庫進行解耦,這樣,程序端只需關(guān)注數(shù)據(jù)庫中間件的地址,而無需知曉底層數(shù)據(jù)庫是如何提供服務(wù)。
作為當前炙手可熱的MySQL中間件,MyCAT實現(xiàn)MySQL主從集群的讀寫分離自是應(yīng)有之義,其配置也相當簡單。
在這里,我用三個實例組成MySQL主從集群,來驗證MyCAT的讀寫分離功能,其實,一主一從就可以滿足,之所以用三個,是為了驗證MyCAT的分片功能。
集群組成如下:
角色 主機名 主機IP
master mysql-server1 192.168.244.145
slave mysql-server2 192.168.244.146
slave mysql-server3 192.168.244.144
在這里,還是使用Travelrecord表進行測試。
首先編輯MyCAT的配置文件schema.xml,關(guān)于dataHost的配置信息如下:
select user()
這里面,有兩個參數(shù)需要注意,balance和 switchType。
其中,balance指的負載均衡類型,目前的取值有4種:
1. balance="0", 不開啟讀寫分離機制,所有讀操作都發(fā)送到當前可用的writeHost上。
2. balance="1",全部的readHost與stand by writeHost參與select語句的負載均衡,簡單的說,當雙主雙從模式(M1->S1,M2->S2,并且M1與 M2互為主備),正常情況下,M2,S1,S2都參與select語句的負載均衡。
3. balance="2",所有讀操作都隨機的在writeHost、readhost上分發(fā)。
4. balance="3",所有讀請求隨機的分發(fā)到wiriterHost對應(yīng)的readhost執(zhí)行,writerHost不負擔讀壓力
switchType指的是切換的模式,目前的取值也有4種:
1. switchType='-1' 表示不自動切換
2. switchType='1' 默認值,表示自動切換
3. switchType='2' 基于MySQL主從同步的狀態(tài)決定是否切換,心跳語句為 show slave status
4. switchType='3'基于MySQL galary cluster的切換機制(適合集群)(1.4.1),心跳語句為 show status like 'wsrep%'。
因此,該配置文件中的balance="1"意味著作為stand by writeHost的hostS1和hostS2將參與select語句的負載均衡,這就實現(xiàn)了主從的讀寫分離,switchType='-1'意味著當主掛掉的時候,不進行自動切換,即hostS1和hostS2并不會被提升為主,仍只提供讀的功能。這就避免了將數(shù)據(jù)寫進slave的可能性,畢竟,單純的MySQL主從集群并不允許將數(shù)據(jù)讀進slave中,除非配置的是雙master。
驗證讀寫分離
下面來驗證一下,
創(chuàng)建Travelrecord表
create table travelrecord (id bigint not null primary key,user_id varchar(100),traveldate DATE, fee decimal,days int);
插入數(shù)據(jù)
mysql> insert into travelrecord(id,user_id,traveldate,fee,days) values(1,@@hostname,20160101,100,10); Query OK, 1 row affected, 1 warning (0.02 sec) mysql> insert into travelrecord(id,user_id,traveldate,fee,days) values(5000001,@@hostname,20160102,100,10); Query OK, 1 row affected, 1 warning (0.01 sec)
在這里,用了一個取巧的方法,即對user_id插入了當前實例的主機名,這樣可直觀的觀察讀寫是否分離以及MyCAT的分片功能。能這樣做的原因在于我當前的MySQL版本-5.6.26默認是基于statement的復制,如果是基于row的復制,則這個方法將不可取。
查詢數(shù)據(jù)
mysql> select * from travelrecord; +---------+---------------+------------+------+------+ | id | user_id | traveldate | fee | days | +---------+---------------+------------+------+------+ | 1 | mysql-server2 | 2016-01-01 | 100 | 10 | | 5000001 | mysql-server3 | 2016-01-02 | 100 | 10 | +---------+---------------+------------+------+------+ rows in set (0.01 sec) mysql> select * from travelrecord; +---------+---------------+------------+------+------+ | id | user_id | traveldate | fee | days | +---------+---------------+------------+------+------+ | 5000001 | mysql-server3 | 2016-01-02 | 100 | 10 | | 1 | mysql-server2 | 2016-01-01 | 100 | 10 | +---------+---------------+------------+------+------+ rows in set (0.02 sec) mysql> select * from travelrecord; +---------+---------------+------------+------+------+ | id | user_id | traveldate | fee | days | +---------+---------------+------------+------+------+ | 5000001 | mysql-server3 | 2016-01-02 | 100 | 10 | | 1 | mysql-server3 | 2016-01-01 | 100 | 10 | +---------+---------------+------------+------+------+ rows in set (0.01 sec) mysql> select * from travelrecord; +---------+---------------+------------+------+------+ | id | user_id | traveldate | fee | days | +---------+---------------+------------+------+------+ | 5000001 | mysql-server3 | 2016-01-02 | 100 | 10 | | 1 | mysql-server3 | 2016-01-01 | 100 | 10 | +---------+---------------+------------+------+------+ rows in set (0.01 sec) mysql> select * from travelrecord; +---------+---------------+------------+------+------+ | id | user_id | traveldate | fee | days | +---------+---------------+------------+------+------+ | 1 | mysql-server2 | 2016-01-01 | 100 | 10 | | 5000001 | mysql-server2 | 2016-01-02 | 100 | 10 | +---------+---------------+------------+------+------+
從上面的輸出結(jié)果,可以得出以下兩點:
一、該配置已實現(xiàn)讀寫分離,讀出來的數(shù)據(jù)沒有master節(jié)點的。
二、MyCAT的隨機分發(fā)不是基于statement的,即一個select語句查詢其中一個節(jié)點,另外一個select語句查詢另外一個節(jié)點。它分發(fā)針對的是片的,同一個select語句的結(jié)果是有不同dataNode返回的。
不僅如此,從MyCAT日志中也可以獲取讀寫分離的相關(guān)信息,當然,前提是MyCAT的日志級別是debug。日志相關(guān)信息如下:
驗證mater掛了,slave還能提供讀的功能
對于MySQL主從集群,我們的需求是master掛了,slave還能提供讀的功能。
下面來測試一下
首先,人為的關(guān)閉主庫
[root@mysql-server1 ~]# /etc/init.d/mysqld stop
登錄MyCAT
[root@mysql-server1 ~]# mysql -utest -ptest -h227.0.0.1 -P8066 -DTESTDB
插入數(shù)據(jù)
mysql> insert into travelrecord(id,user_id,traveldate,fee,days) values(10000001,@@hostname,20160103,100,10); ERROR 1184 (HY000): Connection refused mysql> select * from travelrecord; +---------+---------------+------------+------+------+ | id | user_id | traveldate | fee | days | +---------+---------------+------------+------+------+ | 1 | mysql-server2 | 2016-01-01 | 100 | 10 | | 5000001 | mysql-server3 | 2016-01-02 | 100 | 10 | +---------+---------------+------------+------+------+ rows in set (0.02 sec)
可見無法插入數(shù)據(jù),但不影響讀取數(shù)據(jù)。
至此,MyCAT實現(xiàn)MySQL的讀寫分離部署測試完畢。
總結(jié):
1. 其實,剛開始配置的是readHost節(jié)點,配置如下:
select user()
但這種方式有個問題,即master掛了以后,slave也不能提供服務(wù),而這違反了MySQL主從集群的初衷。
2. 如果開啟了事務(wù)模式,即set autocommit=0,則事務(wù)內(nèi)的讀走的是master節(jié)點,而不是從節(jié)點。