因工作需要,某个项目需要用到ThinkPHP3.1.3进行开发。
因为项目有历史原因,需要连接两个字符编码不同的数据库,一个是UTF8,另一个LATIN1。
用过ThinkPHP的都知道,在/conf/config.php中,找到DB_CHARSET就能设置连接数据库的字符编码。
ThinkPHP默认的字符编码为utf8,可以根据需要修改为LATIN1,GBK,等字符编码。
'DB_CHARSET' => 'utf8'
因为这个项目用到两个数据库,主数据库的charset是utf8,在config.php中配置
//數據庫
'DB_HOST'=>'localhost',
'DB_NAME'=>'main',
'DB_USER'=>'main',
'DB_PWD'=>'mainpassword',
另一个数据库我在config.php中是使用下面的方式配置和调用的。
//数据库配置2
'OTHERDB_CONFIG' => array(
'db_type' => 'mysql',
'db_user' => 'other',
'db_pwd' => 'otherpassword',
'db_host' => '192.168.1.1',
'db_port' => '3306',
'db_name' => 'other',
),
调用方式
$member = M('member', null, 'OTHERDB_CONFIG');
$data = $this->db(1, "OTHERDB_CONFIG")->query("select * from member order by addtime desc");
由于main DB需要使用utf8编码,other DB需要使用LATIN1编码,只有一个DB_CHARSET参数不能满足。
然而我把OTHERDB_CONFIG修改加入db_charset也是无效。
//数据库配置2
'OTHERDB_CONFIG' => array(
'db_type' => 'mysql',
'db_user' => 'other',
'db_pwd' => 'otherpassword',
'db_host' => '192.168.1.1',
'db_port' => '3306',
'db_name' => 'other',
'db_charset' => 'LATIN1' // 加入无效
),
难道ThinkPHP不支持连接多个数据库使用不同的字符编码吗?
查看ThinkPHP源码,找到关于数据库的部分
ThinkPHP/Lib/Driver/Db/DbMysql.class.php
在function connect中找到一句,
mysql_query("SET NAMES '".C('DB_CHARSET')."'", $this->linkID[$linkNum]);
原来ThinkPHP是这样设置连接数据库的字符编码的,居然是直接读取C('DB_CHARSET')的值作为连接数据库的字符编码,并没有提供其他参数设置。
connect 方法是可以传入$config参数的,既然如此,就可以把charset通过$config参数传入,然后修改DbMysql.class.php使其支持根据$config参数设置连接数据库的字符编码了。
因为DbMysql.class.php 是由Db.class.php 调用的,因此$config参数是在Db.class.php中传入。
打开ThinkPHP/Lib/Core/Db.class.php,找到function parseConfig。
$config参数的格式如下:
$db_config = array(
'dbms' => $db_config['db_type'],
'username' => $db_config['db_user'],
'password' => $db_config['db_pwd'],
'hostname' => $db_config['db_host'],
'hostport' => $db_config['db_port'],
'database' => $db_config['db_name'],
'dsn' => $db_config['db_dsn'],
'params' => $db_config['db_params'],
);
因此,我们可以在config.php配置中加入db_params参数,设置db_charset了。
DbMysql.class.php 修改如下, 加入判断$config['params]['db_charset']是否存在,如存在使用参数设置的db_charset否则使用C('DB_CHARSET')
/**
* 连接数据库方法
* @access public
* @throws ThinkExecption
*/
public function connect($config='',$linkNum=0,$force=false) {
if ( !isset($this->linkID[$linkNum]) ) {
if(empty($config)) $config = $this->config;
// 处理不带端口号的socket连接情况
$host = $config['hostname'].($config['hostport']?":{$config['hostport']}":'');
// 是否长连接
$pconnect = !empty($config['params']['persist'])? $config['params']['persist']:$this->pconnect;
if($pconnect) {
$this->linkID[$linkNum] = mysql_pconnect( $host, $config['username'], $config['password'],131072);
}else{
$this->linkID[$linkNum] = mysql_connect( $host, $config['username'], $config['password'],true,131072);
}
if ( !$this->linkID[$linkNum] || (!empty($config['database']) && !mysql_select_db($config['database'], $this->linkID[$linkNum])) ) {
throw_exception(mysql_error());
}
$dbVersion = mysql_get_server_info($this->linkID[$linkNum]);
if(isset($config['params']['db_charset'])){
//使用指定編碼存取数据库
mysql_query("SET NAMES '".$config['params']['db_charset']."'", $this->linkID[$linkNum]);
}else{
//使用UTF8存取数据库
mysql_query("SET NAMES '".C('DB_CHARSET')."'", $this->linkID[$linkNum]);
}
//设置 sql_model
if($dbVersion >'5.0.1'){
mysql_query("SET sql_mode=''",$this->linkID[$linkNum]);
}
// 标记连接成功
$this->connected = true;
// 注销数据库连接配置信息
if(1 != C('DB_DEPLOY_TYPE')) unset($this->config);
}
return $this->linkID[$linkNum];
}
OTHERDB_CONFIG修改如下:
//数据库配置2
'OTHERDB_CONFIG' => array(
'db_type' => 'mysql',
'db_user' => 'other',
'db_pwd' => 'otherpassword',
'db_host' => '192.168.1.1',
'db_port' => '3306',
'db_name' => 'other',
'db_params' => array('db_charset' => 'LATIN1')
),
以后就可以通过db_params设置db_charset了。