CI框架在模型中切换读写库和读写库

时间:2023-03-10 01:40:56
CI框架在模型中切换读写库和读写库

  如果你想在控制器中切换在application/config/database.php中配置好的数据库group,那么你可以参考这篇博客:CI框架在控制器中切换读写库和读写库

   如果你是希望在模型中切换group,那么就不要像在控制器中那样做,比如这样:

<?php
class Hello_model extends CI_Model{
public function getInfo(){
$this->db = null;
$this->load->database("read");
$res = $this->db->query("select * from t1");
print_r($res->result_array());
}
}
?>

  一旦调用Hello_model中的getInfo方法的话,程序就会出错,比如报如下信息的错误:

Message: Call to a member function query() on null

  这个信息无非就是说$this->db是null,切换group失败了。

  咱们来看一下源码是什么样的:位置在system/core/loader.php中,定位到database,如下:

<?php
class CI_Loader {
/**
* @param mixed $params 数据库配置选项
* @param bool $return 是否返回database对象
* @param bool $query_builder 是否开启query builder
*/
public function database($params = '', $return = FALSE, $query_builder = NULL)
{
// 获得CodeIgniter 对象
$CI =& get_instance();
// 检测是否需要加载database对象
if ($return===FALSE && $query_builder===NULL && isset($CI->db) && is_object($CI->db) && ! empty($CI->db->conn_id)){
return FALSE;
}
require_once(BASEPATH.'database/DB.php');
if ($return === TRUE){
return DB($params, $query_builder);
}
//初始化$CI->db,也就是$this->db,防止发生引用错误
$CI->db = '';
// 加载DB类
$CI->db =& DB($params, $query_builder);
return $this;
}
}

  观察执行过程:

    1. Hello_model中的因为$this->db=null,那么执行$this->load->database("read")时,进行到上面的第13行,$CI指向了CodeIgniter对象,
    2. 但是进行if判断时,前面几个都为true,但是isset($CI->db)为false,因为前面已经设为null了,
    3. 所以继续下面的require_once,再进行判断,$return为fase,于是跳过,执行$CI->db = '';
    4. 然后又尝试进行创建db,返回一个引用,注意这里是一个对象实例的引用;而如果上一步的$return为true时,则返回一个db的对象示例。
    5. 最后返回CI_loader本身。也就是说,不管传不传参数$return,都会返回值。

  这个时候,please在Hello_model中尝试打印一下load_database()的返回值

<?php
class Hello_model extends CI_Model{
public function getInfo(){
echo '不传$return,默认为false';
$this->db = null;
print_r($this->load->database("read"));//输出为空 echo '传$return=true';
$this->db = null;
print_r($this->load->database("read",true)); //输出CI_DB_mysqli_driver Object
}
}

  上面代码有两次打印,但是只有第二次打印了一个CI_DB_mysqli_driver Object,至于第一次打印为什么是空,我也没搞明白,也没时间去看了。

  但是查看打印出来的CI_DB_mysqli_driver Object,就会发现,切换到read  group 成功了,也就是说,可以用一个值来接收$this->load->database("read",true)的返回值,

  然后,没错,可以使用$this->db来接收,然后其他的和以前没两样。如下:

<?php
class Hello_model extends CI_Model{ public function getInfo(){
$this->db = null;
$this->db = $this->load->database("read",true); //输出CI_DB_mysqli_driver Object
//使用read group中 exam数据库的tt表
$res = $this->db->query("select * from tt");
print_r($res->result_array());
} public function otherInfo(){
//使用write group中 test数据库的t1表
$res = $this->db->query("select * from t1");
print_r($res->result_array());
}
}