首頁 >後端開發 >php教程 >ThinkPHP框架分散式資料庫連接方法詳解

ThinkPHP框架分散式資料庫連接方法詳解

不言
不言原創
2018-05-03 17:41:432350瀏覽

這篇文章主要介紹了ThinkPHP框架分散式資料庫連接方法,結合實例形式詳細分析了thinkPHP框架針對分散式資料庫的連接方法、操作技巧與相關注意事項,需要的朋友可以參考下

本文實例分析了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,&#39;5.3.6&#39;,&#39;<=&#39;)){
    // 禁用模拟预处理语句
    $this->options[PDO::ATTR_EMULATE_PREPARES] =  false;
  }
  $this->linkID[$linkNum] = new PDO( $config[&#39;dsn&#39;], $config[&#39;username&#39;], $config[&#39;password&#39;],$this->options);
}catch (\PDOException $e) {
  if($autoConnection){ //$autoConnection不为false,而是默认的主服务器
    trace($e->getMessage(),&#39;&#39;,&#39;ERR&#39;);
      return $this->connect($autoConnection,$linkNum); //出现异常,使用递归函数重新连接
    }elseif($config[&#39;debug&#39;]){
      E($e->getMessage());
  }
}

这种方式,对于主从式来说,$r和$m肯定不会相同。因此如果说在读取数据的时候,选择的那台从服务器出现故障的话,那主服务器即是备用的,最后会去主服务器读取。能保证数据读取的时效性。

但是,总感觉现在还不太完善。如果说有多台从服务器,在读取的时候选择的那台从服务器和主服务器都出现了故障,那数据岂不是就读取失败了。这时候如果能再次读取其它的从服务器的话那应该是更有保障。当然了,目前的thinkphp的功能已经相当完善,足够我们使用了。但是还是希望thinkphp以后越来越完善。

相关推荐:

thinkPHP5.0框架命名空间详解




以上是ThinkPHP框架分散式資料庫連接方法詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn