溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

使用ThinkPHP框架怎么實(shí)現(xiàn)分布式數(shù)據(jù)庫(kù)連接

發(fā)布時(shí)間:2021-04-13 16:48:32 來(lái)源:億速云 閱讀:256 作者:Leah 欄目:開發(fā)技術(shù)

使用ThinkPHP框架怎么實(shí)現(xiàn)分布式數(shù)據(jù)庫(kù)連接?相信很多沒(méi)有經(jīng)驗(yàn)的人對(duì)此束手無(wú)策,為此本文總結(jié)了問(wèn)題出現(xiàn)的原因和解決方法,通過(guò)這篇文章希望你能解決這個(gè)問(wèn)題。

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

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

'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è)置完成以后就可以使用了。默認(rèn)情況下就是單一的數(shù)據(jù)庫(kù)連接。

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

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

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

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

下面我們看下面幾個(gè)選項(xiàng)

'DB_HOST'

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

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

對(duì)于這些選項(xiàng)的解析的代碼如下

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

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

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

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

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

'DB_RW_SEPARATE'=>true

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

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

if($this->config['rw_separate']){
   // 主從式采用讀寫分離
   if($master)
     // 主服務(wù)器寫入
     $r =  $m;
   else{
     if(is_numeric($this->config['slave_no'])) {// 指定服務(wù)器讀
       $r = $this->config['slave_no'];
     }else{
        // 讀操作連接從服務(wù)器
       $r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1));  // 每次隨機(jī)連接的數(shù)據(jù)庫(kù)
     }
      }
}else{
   // 讀寫操作不區(qū)分服務(wù)器
   $r = floor(mt_rand(0,count($_config['hostname'])-1));  // 每次隨機(jī)連接的數(shù)據(jù)庫(kù)
}

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

'DB_MASTER_NUM'=>1

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

下面的代碼是選擇主服務(wù)器。

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

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

復(fù)制代碼 代碼如下:

$r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1));   // 每次隨機(jī)連接的數(shù)據(jù)庫(kù)

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

'DB_SLAVE_NO'=> ''

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

if(is_numeric($this->config['slave_no'])) {// 指定服務(wù)器讀
  $r = $this->config['slave_no'];
}else{
  // 讀操作連接從服務(wù)器
  $r = floor(mt_rand($this->config['master_num'],count($_config['hostname'])-1));  // 每次隨機(jī)連接的數(shù)據(jù)庫(kù)
}

以上是對(duì)每個(gè)選項(xiàng)的作用的實(shí)現(xiàn)代碼進(jìn)行了一個(gè)簡(jiǎn)單的說(shuō)明。

下面我們來(lái)看其連接的部分

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);

看到這,我覺得大家應(yīng)該對(duì)上面在介紹各個(gè)配置選項(xiàng)的代碼的時(shí)候其中的$r和$m是什么作用了。

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

connect()函數(shù)的第三個(gè)參數(shù)其實(shí)是表示當(dāng)$db_config這臺(tái)服務(wù)器連接故障時(shí)是否選擇備用的連接。false表示不重連,其它值即表示重新連接。

其核心代碼如下

try{
  if(empty($config['dsn'])) {
   $config['dsn'] =  $this->parseDsn($config);
  }
  if(version_compare(PHP_VERSION,'5.3.6','<=')){
    // 禁用模擬預(yù)處理語(yǔ)句
    $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,而是默認(rèn)的主服務(wù)器
    trace($e->getMessage(),'','ERR');
      return $this->connect($autoConnection,$linkNum); //出現(xiàn)異常,使用遞歸函數(shù)重新連接
    }elseif($config['debug']){
      E($e->getMessage());
  }
}

看完上述內(nèi)容,你們掌握使用ThinkPHP框架怎么實(shí)現(xiàn)分布式數(shù)據(jù)庫(kù)連接的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道,感謝各位的閱讀!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI