ホームページ  >  記事  >  バックエンド開発  >  ThinkPHPフレームワークの分散データベース接続方法を詳しく解説

ThinkPHPフレームワークの分散データベース接続方法を詳しく解説

不言
不言オリジナル
2018-05-03 17:41:432241ブラウズ

この記事は主に ThinkPHP フレームワークの分散データベース接続方法を紹介し、thinkPHP フレームワークの接続方法、分散データベースの操作スキル、および関連する注意事項をサンプルの形式で詳しく分析します。 ThinkPHP フレームワークの分散データベース接続方法の例を分析します。参考のために皆さんと共有してください。詳細は次のとおりです:

Thinkphp は中国で人気のあるフレームワークであり、多くの人が使用しているはずです。この記事では、Thinkphp の重要な部分である分散データベースの接続を分析します。

もちろん、ここではモデルを使用してデータベースを追加、削除、変更、クエリする方法を説明するつもりはありません。私たちは、thinkphp によるデータベースの操作をより深く理解するのに役立つ、基礎となる接続コードの分析を行っています。将来の利用を容易にするため。

1. 単一データベース接続

使用する場合、単一データベースの接続構成は非常に簡単です。構成ファイルにいくつかの情報を構成するだけで済みます。

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

セットアップ完了後、ご利用いただけます。デフォルトは単一のデータベース接続です。

2. 分散データベース接続

単一データベースへの接続は非常に簡単です。分散データベースへの接続を分析してみましょう。

'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'分散データベース。複数のサーバーがある場合は、各アドレスをカンマで区切って複数のサーバーアドレスを入力する必要があります。マスター/スレーブディストリビューションの場合、以前のアドレスはマスター データベースのアドレスである必要があります。

次のユーザー名、パスワード、リスニングポートなどについては、もちろん、持っているだけ書き留めてください。各データベースのユーザー名とパスワードが同じ場合は、1 つのみしか書き込めません。

これらのオプションを解析するコードは次のとおりです

$_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'=>11は分散を意味し、0は集中(つまり、単一サーバー)を意味します。

このオプションの実装はクラス ThinkDbDirver にあります

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 配列内にあります。設定ファイルの解析方法についてはここでは紹介しませんので、興味のある方はThinkDbクラスを参照してください。

$this->multiConnect() 関数が分散接続に使用される場合、'DB_DEPLOY_TYPE' オプションが 1 に設定されている場合、この関数が実行されます。それ以外の場合は、$this->connect() 関数を直接実行します。 $this->config['deploy']表示的就是'DB_DEPLOY_TYPE'这个配置选项,上面的配置在使用之前都已经经过解析了,配置项都在$this->config数组中。至于是如何解析配置文件的,这里我们不做介绍,感兴趣的可以参考ThinkDb类。

$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_RW_SEPARATE'=>true

tru​​eは読み取りと書き込みが分離されていることを意味し、falseは読み取りと書き込みが分離されていないことを意味します。

ここで、読み取りと書き込みの分離はマスター/スレーブ データベース システムに基づいていることに注意してください。このオプションを true に設定すると、マスター データベースが書き込みを行い、スレーブ データベースが読み取りを行います。

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));  // 每次随机连接的数据库
}

$this->config['rw_ Separate'] true の場合、読み取りと書き込みを分離します。 false の場合、読み取りと書き込みを分離しません。なぜ読み取りと書き込みをマスターとスレーブに分離する必要があるのでしょうか?スレーブサーバーは書き込みができず読み取りのみが可能なため、スレーブサーバーにデータを書き込むとデータを同期できません。これにより、データの不整合が発生します。したがって、システムがマスター/スレーブの場合は、読み取りと書き込みを分離する必要があります。つまり、DB_RW_SEPARATE オプションを true に設定する必要があります。

'DB_MASTER_NUM'=>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);
🎜マスター/スレーブ データベースから読み取る場合は、サーバーから読み取ったコア コードを選択します
🎜

コードをコピーします🎜コードは次のとおりです。 🎜

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());
  }
}
🎜 ここで、$this->config['master_num'] はマスター サーバーの数を表します。 🎜🎜🎜🎜'DB_SLAVE_NO'=> ''🎜🎜🎜🎜 データの読み取りに使用されるスレーブ サーバーのシリアル番号を指定します。設定されていない場合、スレーブ サーバーの数はマスター サーバーの数に基づいて計算され、読み取り用にランダムに 1 台が選択されます。 🎜rrreee🎜 以上が各オプションの機能の実装コードの簡単な説明です。 🎜🎜接続部分を見てみましょう🎜rrreee🎜これを見ると、上記の各構成オプションのコードを紹介するときの $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[&#39;dsn&#39;])) {
   $config[&#39;dsn&#39;] =  $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 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。