欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

ThinkPHP框架分布式數(shù)據(jù)庫連接方法詳解

 更新時間:2017年03月14日 11:45:18   作者:chinalorin  
這篇文章主要介紹了ThinkPHP框架分布式數(shù)據(jù)庫連接方法,結合實例形式詳細分析了thinkPHP框架針對分布式數(shù)據(jù)庫的連接方法、操作技巧與相關注意事項,需要的朋友可以參考下

本文實例分析了ThinkPHP框架分布式數(shù)據(jù)庫連接方法。分享給大家供大家參考,具體如下:

Thinkphp作為國內(nèi)的一款流行框架,相信使用的人一定不在少數(shù)。本篇我們來分析一下Thinkphp中比較重要的一部分——分布式數(shù)據(jù)庫的連接。

當然了,我們在這里不是去將如何使用模型去對數(shù)據(jù)庫進行增刪改查。我們是對其底層的連接代碼做一個分析,可以幫助大家更好的理解thinkphp對數(shù)據(jù)庫的操作。方便我們以后的使用。

一、單一數(shù)據(jù)庫的連接

在使用的時候,單一數(shù)據(jù)庫的連接配置非常簡單。我們只需要在配置文件中配置一下的信息即可。

'DB_TYPE' => 'mysql',
'DB_HOST' => '192.168.5.102',
'DB_NAME' => 'databasename',
'DB_USER' => 'user',
'DB_PWD' => 'password',
'DB_PORT' => '3306',
'DB_PREFIX' => 'onmpw_',

設置完成以后就可以使用了。默認情況下就是單一的數(shù)據(jù)庫連接。

二、分布式數(shù)據(jù)庫的連接

單一數(shù)據(jù)庫的連接很簡單,我們著重分析一下分布式數(shù)據(jù)庫的連接。

'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, // 數(shù)據(jù)庫部署方式:0 集中式(單一服務器),1 分布式(主從服務器)
'DB_RW_SEPARATE' => true, // 數(shù)據(jù)庫讀寫是否分離 主從式有效
'DB_MASTER_NUM' => 1, // 讀寫分離后 主服務器數(shù)量
'DB_SLAVE_NO' => '', // 指定從服務器序號

按照以上配置就可以連接分布式數(shù)據(jù)庫了。

下面我們看下面幾個選項

'DB_HOST'

分布式數(shù)據(jù)庫,有幾臺服務器就要填寫幾個服務器地址,每個地址之間用逗號隔開。如果是主從式分布的話,前面的地址要是主數(shù)據(jù)庫的地址。

對于下面的用戶名和密碼還有監(jiān)聽端口之類的,當然是有幾個就寫幾個。如果各個數(shù)據(jù)庫的用戶名和密碼都一樣的話,可以只寫一個。

對于這些選項的解析的代碼如下

$_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 表示的是集中式(也就是單一服務器)。

該選項的實現(xiàn)是在類 Think\Db\Dirver中

protected function initConnect($master=true) {
  if(!empty($this->config['deploy']))
    // 采用分布式數(shù)據(jù)庫
    $this->_linkID = $this->multiConnect($master);
  else
    // 默認單數(shù)據(jù)庫
    if ( !$this->_linkID ) $this->_linkID = $this->connect();
}

$this->config['deploy']表示的就是'DB_DEPLOY_TYPE'這個配置選項,上面的配置在使用之前都已經(jīng)經(jīng)過解析了,配置項都在$this->config數(shù)組中。至于是如何解析配置文件的,這里我們不做介紹,感興趣的可以參考Think\Db類。

$this->multiConnect()函數(shù)就是用來進行分布式連接的,如果'DB_DEPLOY_TYPE'選項設置為1,該函數(shù)就會執(zhí)行。否則直接執(zhí)行$this->connect()函數(shù)。

'DB_RW_SEPARATE'=>true

true 表示讀寫分離;false表示讀寫不分離。

這里需要注意的是,讀寫分離是以主從式數(shù)據(jù)庫系統(tǒng)為前提的。該選項設置為true的時候主數(shù)據(jù)庫寫,從數(shù)據(jù)庫讀。

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));  // 每次隨機連接的數(shù)據(jù)庫
     }
      }
}else{
   // 讀寫操作不區(qū)分服務器
   $r = floor(mt_rand(0,count($_config['hostname'])-1));  // 每次隨機連接的數(shù)據(jù)庫
}

$this->config['rw_separate'] 為true的時候采用讀寫分離,為false的時候讀寫不分離。讀寫分離為什么必須是主從式的呢?因為從服務器不能寫只能讀,如果向從服務器寫入數(shù)據(jù)的話,數(shù)據(jù)是沒法同步的。這樣就會造成數(shù)據(jù)不一致的情況。所以說,如果我們的系統(tǒng)是主從式的話,我們必須采用讀寫分離。也就是說DB_RW_SEPARATE選項必須配置為true。

'DB_MASTER_NUM'=>1

該選項后面的數(shù)字表示讀寫分離后,主服務器的數(shù)量。因此該選項也是用于主從式數(shù)據(jù)庫系統(tǒng)。

下面的代碼是選擇主服務器。

$m = floor(mt_rand(0,$this->config['master_num']-1));

主從式數(shù)據(jù)庫讀取的時候選擇從服務器讀的核心代碼

復制代碼 代碼如下:
$r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1));   // 每次隨機連接的數(shù)據(jù)庫

其中$this->config['master_num']表示主服務器的數(shù)量。

'DB_SLAVE_NO'=> ''

指定從服務器的序號,用于讀取數(shù)據(jù)。如果不設置,則根據(jù)主服務器的數(shù)量計算書從服務器的數(shù)量,然后從中隨機選取一臺進行讀取。

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));  // 每次隨機連接的數(shù)據(jù)庫
}

以上是對每個選項的作用的實現(xiàn)代碼進行了一個簡單的說明。

下面我們來看其連接的部分

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是什么作用了。

現(xiàn)在我們來看 $r == $m ? false : $db_master ,如果數(shù)據(jù)庫讀寫不分離的情況下,讀寫是一臺服務器的話 傳給connect函數(shù)的值為false?;蛘呤侨绻侵鲝姆蛛x的寫的情況下傳給connect的值也為false。通過上面代碼我們看到,如果$r和$m不相等的情況下,會設置$db_master。其實也就是相當于一臺備用的,如果選擇的$r服務器出現(xiàn)故障不能連接,將會去連接$db_master。

connect()函數(shù)的第三個參數(shù)其實是表示當$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); //出現(xiàn)異常,使用遞歸函數(shù)重新連接
    }elseif($config['debug']){
      E($e->getMessage());
  }
}

這種方式,對于主從式來說,$r和$m肯定不會相同。因此如果說在讀取數(shù)據(jù)的時候,選擇的那臺從服務器出現(xiàn)故障的話,那主服務器即是備用的,最后會去主服務器讀取。能保證數(shù)據(jù)讀取的時效性。

但是,總感覺現(xiàn)在還不太完善。如果說有多臺從服務器,在讀取的時候選擇的那臺從服務器和主服務器都出現(xiàn)了故障,那數(shù)據(jù)豈不是就讀取失敗了。這時候如果能再次讀取其它的從服務器的話那應該是更有保障。當然了,目前的thinkphp的功能已經(jīng)相當完善,足夠我們使用了。但是還是希望thinkphp以后越來越完善。

更多關于thinkPHP相關內(nèi)容感興趣的讀者可查看本站專題:《ThinkPHP入門教程》、《thinkPHP模板操作技巧總結》、《ThinkPHP常用方法總結》、《codeigniter入門教程》、《CI(CodeIgniter)框架進階教程》、《Zend FrameWork框架入門教程》、《smarty模板入門基礎教程》及《PHP模板技術總結》。

希望本文所述對大家基于ThinkPHP框架的PHP程序設計有所幫助。

相關文章

  • php獲取訪問者IP地址匯總

    php獲取訪問者IP地址匯總

    在很我的時候我們需要得到用戶的真實IP地址,例如,日志記錄,地理定位,將用戶信息,網(wǎng)站數(shù)據(jù)分析等,其實獲取IP地址很簡單$_SERVER[\'REMOTE_ADDR\']就可以了。下面我們就來給大家匯總一下常用的幾種獲取IP地址的方法。
    2015-04-04
  • PHP實現(xiàn)下載遠程圖片保存到本地的方法

    PHP實現(xiàn)下載遠程圖片保存到本地的方法

    本篇文章主要介紹了PHP實現(xiàn)下載遠程圖片的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-06-06
  • ThinkPHP中的常用查詢語言匯總

    ThinkPHP中的常用查詢語言匯總

    這篇文章主要介紹了ThinkPHP中的常用查詢語言匯總,是ThinkPHP中常用的技巧,在項目開發(fā)中非常有實用價值,需要的朋友可以參考下
    2014-08-08
  • php 分頁類 擴展代碼

    php 分頁類 擴展代碼

    采用url傳遞參數(shù)的方式,可能會有一定的影響或者叫已知的bug,這次做了一些擴展,同時兼容了以前的模式
    2009-06-06
  • Codeigniter(CI)框架分頁函數(shù)及相關知識

    Codeigniter(CI)框架分頁函數(shù)及相關知識

    文章主要介紹了一個自己封裝的Codeigniter(CI)框架的分頁函數(shù)以及Codeigniter(CI)框架分頁類的使用心得,非常簡單實用,希望對大家能有所幫助
    2014-11-11
  • PHP采集騰訊微博的實現(xiàn)代碼

    PHP采集騰訊微博的實現(xiàn)代碼

    PHP采集騰訊微博只留文字部分,大家可以完善它!
    2012-01-01
  • 在php的yii2框架中整合hbase庫的方法

    在php的yii2框架中整合hbase庫的方法

    這篇文章主要介紹了在php的yii2框架中整合hbase庫的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-09-09
  • Thinkphp多文件上傳實現(xiàn)方法

    Thinkphp多文件上傳實現(xiàn)方法

    這篇文章主要介紹了Thinkphp多文件上傳實現(xiàn)方法,是非常實用的技巧,需要的朋友可以參考下
    2014-10-10
  • php/js獲取客戶端mac地址的實現(xiàn)代碼

    php/js獲取客戶端mac地址的實現(xiàn)代碼

    這篇文章主要介紹了如何在php與js中分別獲取客戶度mac地址的方法,需要的朋友可以參考下
    2013-07-07
  • php實現(xiàn)文件下載實例分享

    php實現(xiàn)文件下載實例分享

    php實現(xiàn)對文件的下載需要了解一定的http協(xié)議基礎,php實現(xiàn)下載用到四條http協(xié)議的信息,用header進行會寫處理
    2014-06-06

最新評論