天天看點

分布式資料庫 php,Thinkphp中如何連接配接分布式資料庫

Thinkphp作為國内的主流架構,相信使用的人一定不在少數。今天我們來看一下Thinkphp中如何連接配接分布式資料庫。

當然了,我們在這裡不是去将如何使用模型去對資料庫進行增删改查。我們是對其底層的連接配接代碼做一個分析,可以幫助大家更好的了解thinkphp對資料庫的操作。友善我們以後的使用。

一、單一資料庫的連接配接

在使用的時候,單一資料庫的連接配接配置非常簡單。我們隻需要在配置檔案中配置一下的資訊即可。'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的功能已經相當完善,足夠我們使用了。但是還是希望thinkphp以後越來越完善。

希望上面的介紹能對大家在使用thinkphp做開發的時候有所幫助。

相關推薦: