1.分散式資料庫是什麼:
tp的分散式資料庫主要是通過該配置:
'DB_DEPLOY_TYPE' => 1,// 資料庫部署方式:0 集中式(單一伺服器),1 分布式(主從伺服器)
2.主從伺服器的讀寫分離是什麼:
主從資料庫即為一個主要資料庫會有對應n個從資料庫,而從資料庫只能有一個對應的從資料庫。主從資料庫中寫的操作需要使用主要資料庫,而讀操作使用從資料庫。主要資料庫與從資料庫始終保持資料一致性。其中保持資料庫一致的原理即為當主要資料庫資料發生變化時,會將操作寫入到主要資料庫日誌中,而從資料庫會不停的讀取主要資料庫的日誌儲存到自己的日誌系統中,然後進行執行,從而保持了主從資料庫一致。
3.詳解
一、單一資料庫的串連
在使用的時候,單一資料庫的串連配置非常簡單。我們只需要在設定檔中配置一下的資訊即可。
'DB_TYPE' => 'mysql','DB_HOST' => '192.168.5.102','DB_NAME' => 'databasename','DB_USER' => 'user','DB_PWD' => 'password','DB_PORT' => '3306','DB_PREFIX' => 'onmpw_',
設定完成以後就可以使用了。預設情況下就是單一的資料庫連接。
二、分散式資料庫的串連
單一資料庫的串連很簡單,我們著重分析一下分散式資料庫的串連。
'DB_TYPE' => 'mysql','DB_HOST' => '192.168.5.191,192.168.5.88,192.168.5.103','DB_NAME' => 'test,test,test','DB_USER' => 'masteruser,slaveuser,slaveuser','DB_PWD' => 'masterpass,slavepass,slavepass','DB_PORT' => '3306','DB_PREFIX' => '','DB_DEPLOY_TYPE' => 1, // 資料庫部署方式:0 集中式(單一伺服器),1 分布式(主從伺服器)'DB_RW_SEPARATE' => true, // 資料庫讀寫是否分離 主從式有效'DB_MASTER_NUM' => 1, // 讀寫分離後 主伺服器數量'DB_SLAVE_NO' => '', // 指定從伺服器序號
按照以上配置就可以串連分散式資料庫了。
下面我們看下面幾個選項
‘DB_HOST’
分散式資料庫,有幾台伺服器就要填寫幾個伺服器位址,每個地址之間用逗號隔開。如果是主從式分布的話,前面的地址要是主要資料庫的地址。
對於下面的使用者名稱和密碼還有監聽連接埠之類的,當然是有幾個就寫幾個。如果各個資料庫的使用者名稱和密碼都一樣的話,可以唯寫一個。
對於這些選項的解析的代碼如下
$_config['username'] = explode(',',$this->config['username']);$_config['password'] = explode(',',$this->config['password']);$_config['hostname'] = explode(',',$this->config['hostname']);$_config['hostport'] = explode(',',$this->config['hostport']);$_config['database'] = explode(',',$this->config['database']);$_config['dsn'] = explode(',',$this->config['dsn']);$_config['charset'] = explode(',',$this->config['charset']);
‘DB_DEPLOY_TYPE’=>1
1 表示是分布式, 0 表示的是集中式(也就是單一伺服器)。
該選項的實現是在類 Think\Db\Dirver中
protected function initConnect($master=true) { if(!empty($this->config['deploy'])) // 採用分散式資料庫 $this->_linkID = $this->multiConnect($master); else // 預設單資料庫 if ( !$this->_linkID ) $this->_linkID = $this->connect();}
$this->config[‘deploy’]表示的就是’DB_DEPLOY_TYPE’這個配置選項,上面的配置在使用之前都已經經過解析了,配置項都在$this->config數組中。至於是如何解析設定檔的,這裡我們不做介紹,感興趣的可以參考Think\Db類。
$this->multiConnect()函數就是用來進行分布式串連的,如果’DB_DEPLOY_TYPE’選項設定為1,該函數就會執行。否則直接執行$this->connect()函數。
‘DB_RW_SEPARATE’=>true
true 表示讀寫分離;false表示讀寫不分離。
這裡需要注意的是,讀寫分離是以主從式資料庫系統為前提的。該選項設定為true的時候主要資料庫寫,從資料庫讀。
if($this->config['rw_separate']){ // 主從式採用讀寫分離 if($master) // 主伺服器寫入 $r = $m; else{ if(is_numeric($this->config['slave_no'])) {// 指定伺服器讀 $r = $this->config['slave_no']; }else{ // 讀操作串連從伺服器 $r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1)); // 每次隨機串連的資料庫 } }}else{ // 讀寫操作不區分伺服器 $r = floor(mt_rand(0,count($_config['hostname'])-1)); // 每次隨機串連的資料庫}
$this->config[‘rw_separate’] 為true的時候採用讀寫分離,為false的時候讀寫不分離。讀寫分離為什麼必須是主從式的呢?因為從伺服器不能寫只能讀,如果向從伺服器寫入資料的話,資料是沒法同步的。這樣就會造成資料不一致的情況。所以說,如果我們的系統是主從式的話,我們必須採用讀寫分離。也就是說DB_RW_SEPARATE選項必須配置為true。
'DB_MASTER_NUM'=>1
該選項後面的數字表示讀寫分離後,主伺服器的數量。因此該選項也是用於主從式資料庫系統。
下面的代碼是選擇主伺服器。
$m = floor(mt_rand(0,$this->config['master_num']-1));
主從式資料庫讀取的時候選擇從伺服器讀的核心代碼
$r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1)); // 每次隨機串連的資料庫
其中$this->config[‘master_num’]表示主伺服器的數量。
'DB_SLAVE_NO'=> ''
指定從伺服器的序號,用於讀取資料。如果不設定,則根據主伺服器的數量計算書從伺服器的數量,然後從中隨機選取一台進行讀取。
if(is_numeric($this->config['slave_no'])) {// 指定伺服器讀
$r = $this->config['slave_no'];
}else{
// 讀操作串連從伺服器
$r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1)); // 每次隨機串連的資料庫
}
以上是對每個選項的作用的實現代碼進行了一個簡單的說明。
下面我們來看其串連的部分
if($m != $r ){ $db_master = array( 'username' => isset($_config['username'][$m])?$_config['username'][$m]:$_config['username'][0], 'password' => isset($_config['password'][$m])?$_config['password'][$m]:$_config['password'][0], 'hostname' => isset($_config['hostname'][$m])?$_config['hostname'][$m]:$_config['hostname'][0], 'hostport' => isset($_config['hostport'][$m])?$_config['hostport'][$m]:$_config['hostport'][0], 'database' => isset($_config['database'][$m])?$_config['database'][$m]:$_config['database'][0], 'dsn' => isset($_config['dsn'][$m])?$_config['dsn'][$m]:$_config['dsn'][0], 'charset' => isset($_config['charset'][$m])?$_config['charset'][$m]:$_config['charset'][0], );}$db_config = array( 'username' => isset($_config['username'][$r])?$_config['username'][$r]:$_config['username'][0], 'password' => isset($_config['password'][$r])?$_config['password'][$r]:$_config['password'][0], 'hostname' => isset($_config['hostname'][$r])?$_config['hostname'][$r]:$_config['hostname'][0], 'hostport' => isset($_config['hostport'][$r])?$_config['hostport'][$r]:$_config['hostport'][0], 'database' => isset($_config['database'][$r])?$_config['database'][$r]:$_config['database'][0], 'dsn' => isset($_config['dsn'][$r])?$_config['dsn'][$r]:$_config['dsn'][0], 'charset' => isset($_config['charset'][$r])?$_config['charset'][$r]:$_config['charset'][0],);return $this->connect($db_config,$r,$r == $m ? false : $db_master);
看到這,我覺得大家應該對上面在介紹各個配置選項的代碼的時候其中的$r和$m是什麼作用了。
現在我們來看 $r == $m ? false : $db_master ,如果資料庫讀寫不分離的情況下,讀寫是一台伺服器的話 傳給connect函數的值為false。或者是如果是主從分離的寫的情況下傳給connect的值也為false。通過上面代碼我們看到,如果$r和$m不相等的情況下,會設定$db_master。其實也就是相當於一台備用的,如果選擇的$r伺服器出現故障不能串連,將會去串連$db_master。
connect()函數的第三個參數其實是表示當$db_config這台伺服器串連故障時是否選擇備用的串連。false表示不重連,其它值即表示重新串連。
其核心代碼如下
try{ if(empty($config['dsn'])) { $config['dsn'] = $this->parseDsn($config); } if(version_compare(PHP_VERSION,'5.3.6','<=')){ // 禁用類比預先處理語句 $this->options[PDO::ATTR_EMULATE_PREPARES] = false; } $this->linkID[$linkNum] = new PDO( $config['dsn'], $config['username'], $config['password'],$this->options);}catch (\PDOException $e) { if($autoConnection){ //$autoConnection不為false,而是預設的主伺服器 trace($e->getMessage(),'','ERR'); return $this->connect($autoConnection,$linkNum); //出現異常,使用遞迴函式重新串連 }elseif($config['debug']){ E($e->getMessage()); }}
這種方式,對於主從式來說,$r和$m肯定不會相同。因此如果說在讀取資料的時候,選擇的那台從伺服器出現故障的話,那主伺服器即是備用的,最後會去主伺服器讀取。能保證資料讀取的時效性。
本文講解了thinkphp 分散式資料庫詳解,更多相關內容請關注php中文網。
相關推薦:
如何通過ThinkPHP連結資料庫
如何通過thinkphp串連多資料庫
關於ThinkPHP 5.資料庫的一些基本操作