1、在Db.class.php脚本文件里面的类增加一个魔术方法__get(),写法如下:
public function __get($propertyName)
{ return $this->$propertyName
}
这个方法是用来访问类中protected $config成员属性用的。有的人可能会说,直接把protected改成public岂不是更好。这样只解决了基类的问题,假如,子类也同样进行了受保护,那要你更改更多的文件,这是我们做昌银IT程序员非常不愿意看到的事情。
2、在Model.class.php中的getTableName()方法槐宽更改如下:
$tablepre = $this->db->config['tablepre']
if(empty($this->trueTableName)) {
$tableName??= empty($tablepre) ? $this->tablePrefix : $tablepre
if(!empty($this->tableName)) {
$tableName .= $this->tableName
}
else
{
$tableName .= parse_name($this->name)
}
$this->trueTableName? ? =? ?strtolower($tableName)
}
return (!empty($this->dbName)?$this->dbName.'.':'').$this->
trueTableName这样就完成了多库自由切换时,导致的表前缀问题。
/*******************面向对象PDO连接方式*********************/
'DB_TYPE' =>'PDO', // 数据库类型
'DB_DSN' =>'mysql:host=localhostdbname=master', // DSN连接。
'铅迅亮DB_USER' =>'root', // 数据库用户名
'DB_PWD' =>'123456', // 数据库密码
'DB_PORT' =>'3306', // 数据库端口
'DB_PREFIX' =>'g_', // 数据表前缀
'DB_CHARSET' =>'utf8', // 数据库编码默认采用utf8
select account from db1.test1 unionselect account from db2.test2
union是代表做雹芦神连接两个查询的结果,并自动去重
如果是union all是代表连接两个查询结果,但不自动去重
-----------------------补充---------------------
select a.A,a.B,a.C,b.D from db1.test1 a,db2.test2 b where a.A=b.A
小a小b是把那两个表自纯哗帆定义的缩写,a代表db1.test1表,b代表db2.test2表
本文实例分析了Yii实现MySQL多数据库和读写分离的方法。分享给大家供大家参考。具体分析如下: Yii Framework是一个基于组件、用于开发大型 Web 应用的高性能 PHP 框架。Yii提供了今日Web 2.0应用开发所需要的几乎一切功能,也念腊吵是最强大的框架之一,下文我们来介绍Yii实现MySQL多库和读写分离的方法 前段时间为SNS产品做了架构设计,在程序框架方面做了不少相关的压力测试,最终选定了YiiFramework,至于为什么没选用公司内部的 PHP框架,其实理由很充分,公司的框架虽然是"前辈"们辛苦的积累,但毕竟不够成熟,没有大型项目的历练,犹如一个涉世未深的年轻小伙。Yii作为一个 颇有名气开源产品,必定有很多人在使用,意味着有一批人在维护,而且在这之前,我也使用Yii开发过大型项目,Yii的设计模式和它的易扩展特性足以堪当重任。 SNS同一般的社交产品不同的就是它最终要承受大并发和大数据量的考验,架构设计时就要考虑这些问题, web分布式、负载均衡、分布式文件存储、仔侍MySQL分布式或读写分离、NoSQL以及各种缓存,这些都是必不可少的应用方案,局李本文所讲的就是MySQL 分库和主从读写分离在Yii的配置和使用。 Yii默认是不支持读写分离的,我们可以利用Yii的事件驱动模式来实现MySQL的读写分离。 Yii提供了一个强大的CActiveRecord数据库 *** 作类,通过重写getDbConnection方法来实现数据库的切换,然后通过事件 beforeSave、beforeDelete、beforeFind 来实现读写服务器的切换,还需要两个配置文件dbconfig和modelconfig分别配置数据库主从服务器和model所对应的数据库名称,附代码 DBConfig.php文件如下: 复制代码 代码如下:<?php return array( 'passport' =>array( 'write' =>array( 'class' =>'CDbConnection', 'connectionString' =>'mysql:host=10.1.39.2dbname=db1′, 'emulatePrepare' =>true, //'enableParamLogging' =>true, 'enableProfiling' =>true, 'username' =>'root', 'password' =>'', 'charset' =>'utf8′, 'schemaCachingDuration'=>3600, ), 'read' =>array( array( 'class' =>'CDbConnection', 'connectionString' =>'mysql:host=10.1.39.3dbname=db1, 'emulatePrepare' =>true, //'enableParamLogging' =>true, 'enableProfiling' =>true, 'username' =>'root', 'password' =>'', 'charset' =>'utf8′, 'schemaCachingDuration'=>3600, ), array( 'class' =>'CDbConnection', 'connectionString' =>'mysql:host=10.1.39.4dbname=db3′, 'emulatePrepare' =>true, //'enableParamLogging' =>true, 'enableProfiling' =>true, 'username' =>'root', 'password' =>'', 'charset' =>'utf8′, 'schemaCachingDuration'=>3600, ), ), ), ) ModelConfig.php如下: 复制代码 代码如下:<?php return array( //key为数据库名称,value为Model 'passport' =>array('User','Post'), 'microblog' =>array('…'), ) ?> ActiveRecord.php如下: 复制代码 代码如下:/** * 基于CActiveRecord类的封装,实现多库和主从读写分离 * 所有Model都必须继承些类. * */ class ActiveRecord extends CActiveRecord { //model配置 public $modelConfig = '' //数据库配置 public $dbConfig = '' //定义一个多数据库集合 static $dataBase = array() //当前数据库名称 public $dbName = '' //定义库类型(读或写) public $dbType = 'read'//'read' or 'write' /** * 在原有基础上添加了一个dbname参数 * @param string $scenario Model的应用场景 * @param string $dbname 数据库名称 */ public function __construct($scenario='insert', $dbname = '') { if (!empty($dbname)) $this->dbName = $dbname parent::__construct($scenario) } /** * 重写父类的getDbConnection方法 * 多库和主从都在这里切换 */ public function getDbConnection() { //如果指定的数据库对象存在则直接返回 if (self::$dataBase[$this->dbName]!==null) return self::$dataBase[$this->dbName] if ($this->dbName == 'db'){ self::$dataBase[$this->dbName] = Yii::app()->getDb() }else{ $this->changeConn($this->dbType) } if(self::$dataBase[$this->dbName] instanceof CDbConnection){ self::$dataBase[$this->dbName]->setActive(true) return self::$dataBase[$this->dbName] } else throw new CDbException(Yii::t('yii','Model requires a "db" CDbConnection application component.')) } /** * 获取配置文件 * @param unknown_type $type * @param unknown_type $key */ private function getConfig($type="modelConfig",$key="){ $config = Yii::app()->params[$type] if($key) $config = $config[$key] return $config } /** * 获取数据库名称 */ private function getDbName(){ if($this->dbName) return $this->dbName $modelName = get_class($this->model()) $this->modelConfig = $this->getConfig('modelConfig') //获取model所对应的数据库名 if($this->modelConfig)foreach($this->modelConfig as $key=>$val){ if(in_array($modelName,$val)){ $dbName = $key break } } return $dbName } /** * 切换数据库连接 * @param unknown_type $dbtype */ protected function changeConn($dbtype = 'read'){ if($this->dbType == $dbtype &&self::$dataBase[$this->dbName] !== null) return self::$dataBase[$this->dbName] $this->dbName = $this->getDbName() if(Yii::app()->getComponent($this->dbName.'_'.$dbtype) !== null){ self::$dataBase[$this->dbName] = Yii::app()->getComponent($this->dbName.'_'.$dbtype) return self::$dataBase[$this->dbName] } $this->dbConfig = $this->getConfig('dbConfig',$this->dbName) //跟据类型取对应的配置(从库是随机值) if($dbtype == 'write'){ $config = $this->dbConfig[$dbtype] }else{ $slavekey = array_rand($this->dbConfig[$dbtype]) $config = $this->dbConfig[$dbtype][$slavekey] } //将数据库配置加到component中 if($dbComponent = Yii::createComponent($config)){ Yii::app()->setComponent($this->dbName.'_'.$dbtype,$dbComponent) self::$dataBase[$this->dbName] = Yii::app()->getComponent($this->dbName.'_'.$dbtype) $this->dbType = $dbtype return self::$dataBase[$this->dbName] } else throw new CDbException(Yii::t('yii','Model requires a "changeConn" CDbConnection application component.')) } /** * 保存数据前选择 主 数据库 */ protected function beforeSave(){ parent::beforeSave() $this->changeConn('write') return true } /** * 删除数据前选择 主 数据库 */ protected function beforeDelete(){ parent::beforeDelete() $this->changeConn('write') return true } /** * 读取数据选择 从 数据库 */ protected function beforeFind(){ parent::beforeFind() $this->changeConn('read') return true } /** * 获取master库对象 */ public function dbWrite(){ return $this->changeConn('write') } /** * 获取slave库对象 */ public function dbRead(){ return $this->changeConn('read') } } 这是我写好的类,放在components文件夹里,然后所有的Model都继承ActiveRecord类就可以实现多库和主从读写分离了,至于如何支持原生的SQL也同时使用读写分离,此类都已经实现。 希望本文所述对大家基于Yii框架的PHP程序设计有所帮助。欢迎分享,转载请注明来源:内存溢出
评论列表(0条)