经过前面几节的内容,咱们从零开始搭建了一个很是方便的MVC框架,理解了 CodeIgniter 框架最核心的部分。然而一个框架的便利不单单在于提供一个MVC就能够了,它还必须具备较高的扩展性。下面将从 CodeIgniter 的官方文档中的顺序一步一步充实咱们这个“丑陋” 的框架。php
本课将实现 CodeIgniter 类库加载的模式,对应于http://codeigniter.org.cn/user_guide/general/libraries.htmlhtml
首先类库加载设计在 Loader 类中,函数命名为 library,以下所示数组
public function library($library = '', $params = NULL, $object_name = NULL) { if (is_array($library)) { foreach ($library as $class) { $this->library($class, $params); } return; } if ($library == '' OR isset($this->_base_classes[$library])) { return FALSE; } if ( ! is_null($params) && ! is_array($params)) { $params = NULL; } $this->_ci_load_class($library, $params, $object_name); }
与 model 相似,它首先判断传入的第一个参数是否为数组,若是是的话,就递归调用一次。app
它经过调用 _ci_load_class 函数来实际加载具体的类。框架
protected function _ci_load_class($class, $params = NULL, $object_name = NULL) { $class = str_replace('.php', '', trim($class, '/')); $subdir = ''; if (($last_slash = strrpos($class, '/')) !== FALSE) { // 提取出路径 $subdir = substr($class, 0, $last_slash + 1); // 提取出 class $class = substr($class, $last_slash + 1); } foreach (array(ucfirst($class), strtolower($class)) as $class) { $subclass = APPPATH.'libraries/'.$subdir.config_item('subclass_prefix').$class.'.php'; // class 扩展 if (file_exists($subclass)) { $baseclass = BASEPATH.'libraries/'.ucfirst($class).'.php'; if ( ! file_exists($baseclass)) { log_message('error', 'Unable to load the requested class: '.$class); show_error('Unable to load the requested class: '.$class); } // 检查 class 是否加载过 if (in_array($subclass, $this->_ci_loaded_files)) { if (! is_null($object_name)) { $CI =& get_instance(); if ( ! isset($CI->$object_name)) { return $this->_ci_init_class($class, config_item('subclass_prefix'), $params, $object_name); } } $is_duplicate = TRUE; log_message('debug', $class." class already loaded. Second attempt ignored."); return; } include_once($baseclass); include_once($subclass); $this->_ci_loaded_files[] = $subclass; return $this->_ci_init_class($class, config_item('subclass_prefix'), $params, $object_name); } } if ($is_duplicate == FALSE) { log_message('error', 'Unable to load the requested class: '.$class); show_error("Unable to load the requested class: ".$class); } }
按照 CodeIgniter 的思路,首先会对加载函数的第一个参数路径进行拆分,获得具体的路径和文件名,由于文件名会有一些特殊的约定。ide
CodeIgniter 在设计的时候有个很重要的思想,就是区分 system 和 application 两种类型的代码,从而设计出灵活性很高的框架,那么针对 类库 而言,就是在 system 中提供一些经常使用的类库,而后给用户在这个基础上扩展或覆盖的权利,另外,也能够直接在 函数
application 文件夹下设计出彻底自定义的类。codeigniter
本节主要实现 在原有类上进行扩展 的代码ui
1) 首先判断 application 文件夹下是否存在按照约定命名的扩展类库文件,即 subclass_prefix = 'MY_', 就判断是否存在 MY_类库名.php 文件this
2)若是存在,则加载基类库文件,所有 include 以后,就能够 _ci_init_class 了
public function _ci_init_class($class, $prefix = '', $config = FALSE, $object_name = NULL) { if ($prefix == '') { $name = $class; } else { $name = $prefix.$class; } if ( ! class_exists($name)) { log_message('error', "Non-existent class: ".$name); show_error("Non-existent class: ".$class); } $class = strtolower($class); $classvar = $class; $CI =& get_instance(); if ($config !== NULL) { $CI->$classvar = new $name($config); } else { $CI->$classvar = new $name; } }
而后经过
$CI->$classvar = new $name($config);
成为 CI 核心对象的成员,从而就能够很方便的访问了~ $this->email->函数名();