ThinkPHP框架分布式數(shù)據(jù)庫(kù)連接方法詳解
本文實(shí)例分析了ThinkPHP框架分布式數(shù)據(jù)庫(kù)連接方法。分享給大家供大家參考,具體如下:
Thinkphp作為國(guó)內(nèi)的一款流行框架,相信使用的人一定不在少數(shù)。本篇我們來分析一下Thinkphp中比較重要的一部分——分布式數(shù)據(jù)庫(kù)的連接。
當(dāng)然了,我們?cè)谶@里不是去將如何使用模型去對(duì)數(shù)據(jù)庫(kù)進(jìn)行增刪改查。我們是對(duì)其底層的連接代碼做一個(gè)分析,可以幫助大家更好的理解thinkphp對(duì)數(shù)據(jù)庫(kù)的操作。方便我們以后的使用。
一、單一數(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)過解析了,配置項(xiàng)都在$this->config
數(shù)組中。至于是如何解析配置文件的,這里我們不做介紹,感興趣的可以參考Think\Db類。
$this->multiConnect()
函數(shù)就是用來進(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ù)是沒法同步的。這樣就會(huì)造成數(shù)據(jù)不一致的情況。所以說,如果我們的系統(tǒng)是主從式的話,我們必須采用讀寫分離。也就是說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ù)器讀的核心代碼
其中$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)單的說明。
下面我們來看其連接的部分
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)在我們來看 $r == $m ? false : $db_master ,如果數(shù)據(jù)庫(kù)讀寫不分離的情況下,讀寫是一臺(tái)服務(wù)器的話 傳給connect函數(shù)的值為false?;蛘呤侨绻侵鲝姆蛛x的寫的情況下傳給connect的值也為false。通過上面代碼我們看到,如果$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ù)處理語句 $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()); } }
這種方式,對(duì)于主從式來說,$r和$m肯定不會(huì)相同。因此如果說在讀取數(shù)據(jù)的時(shí)候,選擇的那臺(tái)從服務(wù)器出現(xiàn)故障的話,那主服務(wù)器即是備用的,最后會(huì)去主服務(wù)器讀取。能保證數(shù)據(jù)讀取的時(shí)效性。
但是,總感覺現(xiàn)在還不太完善。如果說有多臺(tái)從服務(wù)器,在讀取的時(shí)候選擇的那臺(tái)從服務(wù)器和主服務(wù)器都出現(xiàn)了故障,那數(shù)據(jù)豈不是就讀取失敗了。這時(shí)候如果能再次讀取其它的從服務(wù)器的話那應(yīng)該是更有保障。當(dāng)然了,目前的thinkphp的功能已經(jīng)相當(dāng)完善,足夠我們使用了。但是還是希望thinkphp以后越來越完善。
更多關(guān)于thinkPHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《ThinkPHP入門教程》、《thinkPHP模板操作技巧總結(jié)》、《ThinkPHP常用方法總結(jié)》、《codeigniter入門教程》、《CI(CodeIgniter)框架進(jìn)階教程》、《Zend FrameWork框架入門教程》、《smarty模板入門基礎(chǔ)教程》及《PHP模板技術(shù)總結(jié)》。
希望本文所述對(duì)大家基于ThinkPHP框架的PHP程序設(shè)計(jì)有所幫助。
- ThinkPHP實(shí)現(xiàn)多數(shù)據(jù)庫(kù)連接的解決方法
- ThinkPHP 連接Oracle數(shù)據(jù)庫(kù)的詳細(xì)教程[全]
- ThinkPHP連接數(shù)據(jù)庫(kù)及主從數(shù)據(jù)庫(kù)的設(shè)置教程
- thinkphp配置連接數(shù)據(jù)庫(kù)技巧
- ThinkPHP連接數(shù)據(jù)庫(kù)的方式匯總
- thinkPHP連接sqlite3數(shù)據(jù)庫(kù)的實(shí)現(xiàn)方法(附Thinkphp代碼生成器下載)
- ThinkPHP框架基于PDO方式連接數(shù)據(jù)庫(kù)操作示例
- tp5(thinkPHP5)框架連接數(shù)據(jù)庫(kù)的方法示例
- ThinkPHP連接Oracle數(shù)據(jù)庫(kù)
- thinkphp3.2同時(shí)連接兩個(gè)數(shù)據(jù)庫(kù)的簡(jiǎn)單方法
相關(guān)文章
PHP實(shí)現(xiàn)下載遠(yuǎn)程圖片保存到本地的方法
本篇文章主要介紹了PHP實(shí)現(xiàn)下載遠(yuǎn)程圖片的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-06-06Codeigniter(CI)框架分頁(yè)函數(shù)及相關(guān)知識(shí)
文章主要介紹了一個(gè)自己封裝的Codeigniter(CI)框架的分頁(yè)函數(shù)以及Codeigniter(CI)框架分頁(yè)類的使用心得,非常簡(jiǎn)單實(shí)用,希望對(duì)大家能有所幫助2014-11-11php/js獲取客戶端mac地址的實(shí)現(xiàn)代碼
這篇文章主要介紹了如何在php與js中分別獲取客戶度mac地址的方法,需要的朋友可以參考下2013-07-07