Yii 实现MySQL多库和读写分离详解

jerry Yii 2015年08月23日 收藏

DBConfig.php

  1. <?php
  2. return array(
  3.     'passport' => array(
  4.         'write' => array(
  5.             'class' => 'CDbConnection',
  6.             'connectionString' => 'mysql:host=10.1.39.2;dbname=db1',
  7.             'emulatePrepare' => true,
  8.             //'enableParamLogging' => true,
  9.             'enableProfiling' => true,
  10.             'username' => 'root',
  11.             'password' => ",
  12.             'charset' => 'utf8',
  13.             'schemaCachingDuration'=>3600,
  14.         ),
  15.         'read' => array(
  16.             array(
  17.                 'class' => 'CDbConnection',
  18.                 'connectionString' => 'mysql:host=10.1.39.3;dbname=db1',
  19.                 'emulatePrepare' => true,
  20.                 //'enableParamLogging' => true,
  21.                 'enableProfiling' => true,
  22.                 'username' => 'root',
  23.                 'password' => ",
  24.                 'charset' => 'utf8',
  25.                 'schemaCachingDuration'=>3600,
  26.             ),
  27.             array(
  28.                 'class' => 'CDbConnection',
  29.                 'connectionString' => 'mysql:host=10.1.39.4;dbname=db3',
  30.                 'emulatePrepare' => true,
  31.                 //'enableParamLogging' => true,
  32.                 'enableProfiling' => true,
  33.                 'username' => 'root',
  34.                 'password' => ",
  35.                 'charset' => 'utf8',
  36.                 'schemaCachingDuration'=>3600,
  37.             ),
  38.         ),
  39.     ),
  40. );

ModelConfig.php

  1. <?php
  2. return array(
  3.     //key为数据库名称,value为Model
  4.     'passport' => array('User','Post'),
  5.     'microblog' => array('…'),
  6. );
  7.     
  8. ?>

ActiveRecord.php

  1. /**
  2. * 基于CActiveRecord类的封装,实现多库和主从读写分离
  3. * 所有Model都必须继承些类.
  4. *
  5. * @author atao<lnbalife@126.com>
  6. */
  7. class ActiveRecord extends CActiveRecord
  8. {
  9.     //model配置
  10.     public $modelConfig = '';
  11.     
  12.     //数据库配置
  13.     public $dbConfig = '';
  14.     
  15.     //定义一个多数据库集合
  16.     static $dataBase = array();
  17.     
  18.     //当前数据库名称
  19.     public $dbName = '';
  20.     
  21.     //定义库类型(读或写)
  22.     public $dbType = 'read'; //'read' or 'write'
  23.     
  24.     /**
  25.      * 在原有基础上添加了一个dbname参数
  26.      * @param string $scenario Model的应用场景
  27.      * @param string $dbname 数据库名称
  28.      */
  29.     public function __construct($scenario='insert', $dbname = '')
  30.     {
  31.         if (!empty($dbname))
  32.             $this->dbName = $dbname;
  33.         parent::__construct($scenario);
  34.     }
  35.     
  36.     /**
  37.      * 重写父类的getDbConnection方法
  38.      * 多库和主从都在这里切换
  39.      */
  40.     public function getDbConnection()
  41.     {
  42.         //如果指定的数据库对象存在则直接返回
  43.         if (self::$dataBase[$this->dbName]!==null)
  44.             return self::$dataBase[$this->dbName];
  45.     
  46.         if ($this->dbName == 'db'){
  47.             self::$dataBase[$this->dbName] = Yii::app()->getDb();
  48.         }else{
  49.             $this->changeConn($this->dbType);
  50.         }
  51.     
  52.         if(self::$dataBase[$this->dbName] instanceof CDbConnection){
  53.             self::$dataBase[$this->dbName]->setActive(true);
  54.             return self::$dataBase[$this->dbName];
  55.         } else
  56.             throw new CDbException(Yii::t('yii','Model requires a "db" CDbConnection application component.'));
  57.     }
  58.     
  59.     /**
  60.      * 获取配置文件
  61.      * @param unknown_type $type
  62.      * @param unknown_type $key
  63.      */
  64.     private function getConfig($type="modelConfig",$key=''){
  65.         $config = Yii::app()->params[$type];
  66.         if($key)
  67.             $config = $config[$key];
  68.         return $config;
  69.     }
  70.     
  71.     /**
  72.      * 获取数据库名称
  73.      */
  74.     private function getDbName(){
  75.         if($this->dbName)
  76.             return $this->dbName;
  77.         $modelName = get_class($this->model());
  78.         $this->modelConfig = $this->getConfig('modelConfig');
  79.         //获取model所对应的数据库名
  80.         if($this->modelConfig) foreach($this->modelConfig as $key=>$val){
  81.             if(in_array($modelName,$val)){
  82.                 $dbName = $key;
  83.                 break;
  84.             }
  85.         }
  86.         return $dbName;
  87.     }
  88.     
  89.     /**
  90.      * 切换数据库连接
  91.      * @param unknown_type $dbtype
  92.      */
  93.     protected function changeConn($dbtype = 'read'){
  94.         if($this->dbType == $dbtype && self::$dataBase[$this->dbName] !== null)
  95.             return self::$dataBase[$this->dbName];
  96.     
  97.         $this->dbName = $this->getDbName();
  98.     
  99.         if(Yii::app()->getComponent($this->dbName.'_'.$dbtype) !== null){
  100.             self::$dataBase[$this->dbName] = Yii::app()->getComponent($this->dbName.'_'.$dbtype);
  101.             return self::$dataBase[$this->dbName];
  102.         }
  103.     
  104.         $this->dbConfig = $this->getConfig('dbConfig',$this->dbName);
  105.     
  106.         //跟据类型取对应的配置(从库是随机值)
  107.         if($dbtype == 'write'){
  108.             $config = $this->dbConfig[$dbtype];
  109.         }else{
  110.             $slavekey = array_rand($this->dbConfig[$dbtype]);
  111.             $config = $this->dbConfig[$dbtype][$slavekey];
  112.         }
  113.     
  114.         //将数据库配置加到component中
  115.         if($dbComponent = Yii::createComponent($config)){
  116.             Yii::app()->setComponent($this->dbName.'_'.$dbtype,$dbComponent);
  117.             self::$dataBase[$this->dbName] = Yii::app()->getComponent($this->dbName.'_'.$dbtype);
  118.             $this->dbType = $dbtype;
  119.             return self::$dataBase[$this->dbName];
  120.         } else
  121.             throw new CDbException(Yii::t('yii','Model requires a "changeConn" CDbConnection application component.'));
  122.     }
  123.     
  124.     /**
  125.      * 保存数据前选择 主 数据库
  126.      */
  127.     protected function beforeSave(){
  128.         parent::beforeSave();
  129.         $this->changeConn('write');
  130.         return true;
  131.     }
  132.     
  133.     /**
  134.      * 删除数据前选择 主 数据库
  135.      */
  136.     protected function beforeDelete(){
  137.         parent::beforeDelete();
  138.         $this->changeConn('write');
  139.         return true;
  140.     }
  141.     
  142.     /**
  143.      * 读取数据选择 从 数据库
  144.      */
  145.     protected function beforeFind(){
  146.         parent::beforeFind();
  147.         $this->changeConn('read');
  148.         return true;
  149.     }
  150.     
  151.     /**
  152.      * 获取master库对象
  153.      */
  154.     public function dbWrite(){
  155.         return $this->changeConn('write');
  156.     }
  157.     
  158.     /**
  159.      * 获取slave库对象
  160.      */
  161.     public function dbRead(){
  162.         return $this->changeConn('read');
  163.     }
  164. }