這篇文章主要介紹了PHP CodeIgniter框架的工作原理研究,本文首先分析了它的工作流程,然后總結(jié)了它的工作原理,需要的朋友可以參考下
CodeIgniter(以下簡稱CI,官網(wǎng)以及中國站)是一個(gè)流行的PHP框架,小巧但功能強(qiáng)大,簡潔輕量同時(shí)擁有很好的擴(kuò)展性,在國內(nèi)也比較受歡迎。另一方面,CI卻沒有與時(shí)俱進(jìn),并不支持PHP5.3之后的一些特性,導(dǎo)致它相對更適合較老一些的項(xiàng)目。雖然如此,CI仍是一個(gè)優(yōu)秀的框架,而且它本身內(nèi)核較小,源碼優(yōu)雅,適于學(xué)習(xí)。
CI易于使用,可以方便的開發(fā)出web應(yīng)用。先來看一下CI的工作流程圖(此處內(nèi)容引用自)
PHP CodeIgniter框架的工作原理研究 三聯(lián)、
1.index.php 作為前端控制器,初始化運(yùn)行 CodeIgniter 所需要的基本資源。
2.Router 檢查 HTTP 請求,以確定誰來處理請求。
3.如果緩存(Cache)文件存在,它將繞過通常的系統(tǒng)執(zhí)行順序,被直接發(fā)送給瀏覽器。
4.安全(Security)。應(yīng)用程序控制器(Application Controller)裝載之前,HTTP 請求和任何用戶提交的數(shù)據(jù)將被過濾。
5.控制器(Controller)裝載模型、核心庫、輔助函數(shù),以及任何處理特定請求所需的其它資源。
6.最終視圖(View)渲染發(fā)送到 Web 瀏覽器中的內(nèi)容。如果開啟緩存(Caching),視圖首先被緩存,所以將可用于以后的請求。
以上給出了一個(gè)大致流程。那么當(dāng)看到頁面在瀏覽器中呈現(xiàn)時(shí),程序內(nèi)部究竟是如何工作的呢?
下面按照執(zhí)行順序,依次列出了CI框架主要加載的文件,并簡要介紹其作用:
01. index.php
定義使用環(huán)境(ENVIRONMENT),框架路徑(system_path,BASEPATH),應(yīng)用目錄(application_folder),應(yīng)用路徑(APPPATH)等,加載(require)CI核心文件
02. BASEPATH/core/CodeIgniter.php (ps.實(shí)際上BASEPATH包含最后的文件分隔符'/',這里額外加上了'/'是為了更清晰的展示)
系統(tǒng)初始化文件,整個(gè)框架最核心的部分,在此加載(load)了一系列的base class,并且執(zhí)行這次請求
03. BASEPATH/core/Common.php
common文件包含一系列的基礎(chǔ)和公共函數(shù) ,供全局使用,例如load_class(),get_config()等
04. BASEPATH/core/Benchmark
這是一個(gè)基準(zhǔn)測試類,默認(rèn)標(biāo)注了應(yīng)用各個(gè)階段的執(zhí)行點(diǎn),以得到其執(zhí)行時(shí)間。也允許你自己定義監(jiān)測點(diǎn)。
05. BASEPATH/core/Hooks.php
CI_Hooks是一個(gè)鉤子類,是框架進(jìn)行擴(kuò)展的核心,能夠在程序允許的各個(gè)階段插入掛鉤點(diǎn),執(zhí)行你自定義的類,函數(shù)等
06. BASEPATH/core/Config.php
配置文件管理類,加載讀取或設(shè)置配置
07. BASEPATH/core/URI.php, BASEPATH/core/Router.php
URI類幫助你解析請求的uri,并提供分割uri的函數(shù)集合,供Router類使用
08. BASEPATH/core/Router.php
路由類,即通過請求的uri,和用戶配置的路由(APPPATH/config/routes.php),將用戶請求分發(fā)到指定的處理函數(shù)中(通常來說是某一個(gè)Controller實(shí)例中某一action函數(shù))
09. BASEPATH/core/Output.php, BASEPATH/core/Input.php
輸入類,即處理請求的輸入?yún)?shù),提供安全的獲取方式。輸出類將最后的執(zhí)行結(jié)果發(fā)送出去,它還負(fù)責(zé)緩存的功能
10. BASEPATH/core/Controller.php
控制器基類,用單例模式對外提供實(shí)例,整個(gè)應(yīng)用程序的心臟。它是一個(gè)Super Object,在應(yīng)用內(nèi)加載的類都可以成為控制器的成員變量,這一點(diǎn)非常重要,會(huì)在之后繼續(xù) 講到。
11. APPPATH/controllers/$RTR->fetch_directory().$RTR->fetch_class().'.php'
通過路由功能,得到控制器名,實(shí)例化真正的控制器類(子類)
12. BASEPATH/core/Loader.php
CI_Loader用于加載應(yīng)用程序中的各種類庫,模型,視圖,數(shù)據(jù)庫,文件等,并設(shè)置成為控制器的成員變量
13. call_user_func_array 調(diào)用處理函數(shù)
通過路由,得到action函數(shù)名,調(diào)用 Controller->action()函數(shù),處理應(yīng)用邏輯,實(shí)際業(yè)務(wù)處理邏輯便是在action函數(shù)中寫的
14. $OUT->_display() 將內(nèi)容輸出
以上便是整個(gè)應(yīng)用程序最基礎(chǔ)的處理流程。下面選取核心內(nèi)容代碼再進(jìn)行說明,以加強(qiáng)對CI的理解:
<?php
//*BASEPATH/system/core/Common.php
//引導(dǎo)文件中Benchmark,Hooks,Config等都是通過這個(gè)函數(shù)進(jìn)行加載的
function &load_class($class, $directory = 'libraries', $prefix = 'CI_')
{
//記錄加載過的類
static $_classes = array();
// 已經(jīng)加載過,直接讀取并返回
if (isset($_classes[$class]))
{
return $_classes[$class];
}
$name = FALSE;
// 在指定目錄尋找要加載的類
foreach (array(APPPATH, BASEPATH) as $path)
{
if (file_exists($path.$directory.'/'.$class.'.php'))
{
$name = $prefix.$class;
if (class_exists($name) === FALSE)
{
require($path.$directory.'/'.$class.'.php');
}
break;
}
}
// 沒有找到
if ($name === FALSE)
{
exit('Unable to locate the specified class: '.$class.'.php');
}
// 追蹤記錄下剛才加載的類,is_loaded()函數(shù)在下面
is_loaded($class);
$_classes[$class] = new $name();
return $_classes[$class];
}
// 記錄已經(jīng)加載過的類。函數(shù)返回所有加載過的類
function &is_loaded($class = '')
{
static $_is_loaded = array();
if ($class != '')
{
$_is_loaded[strtolower($class)] = $class;
}
return $_is_loaded;
}
//*BASEPATH/system/core/Controller.php
class CI_Controller {
private static $instance;
public function __construct()
{
self::$instance =& $this;
//將所有在引導(dǎo)文件中(CodeIgniter.php)初始化的類對象(即剛才4,5,6,7,8,9等步驟),
//注冊成為控制器類的成員變量,就使得這個(gè)控制器成為一個(gè)超級對象(super object)
foreach (is_loaded() as $var => $class)
{
$this->$var =& load_class($class);
}
<span style="white-space:pre"> </span>//加載Loader對象,再利用Loader對象對程序內(nèi)一系列資源進(jìn)行加載<span style="white-space:pre"> </span>
$this->load =& load_class('Loader', 'core');
$this->load->initialize();
log_message('debug', "Controller Class Initialized");
}
//這個(gè)函數(shù)對外提供了控制器的單一實(shí)例
public static function &get_instance()
{
return self::$instance;
}
}
//*BASEPATH/system/core/CodeIgniter.php
// Load the base controller class
require BASEPATH.'core/Controller.php';
//通過這個(gè)全局函數(shù)就得到了控制器的實(shí)例,得到了這個(gè)超級對象,
//意味著在程序其他地方調(diào)用這個(gè)函數(shù),就能得到整個(gè)框架的控制權(quán)
function &get_instance()
{
return CI_Controller::get_instance();
}
// 加載對應(yīng)的控制器類
// 注意:Router類會(huì)自動(dòng)使用 router->_validate_request() 驗(yàn)證控制器路徑
if ( ! file_exists(APPPATH.'controllers/'.$RTR->fetch_directory().$RTR->fetch_class().'.php'))
{
show_error('Unable to load your default controller. Please make sure the controller specified in your Routes.php file is valid.');
}
include(APPPATH.'controllers/'.$RTR->fetch_directory().$RTR->fetch_class().'.php');
$class = $RTR->fetch_class(); //Controller class name
$method = $RTR->fetch_method(); //action name
//.....
// 調(diào)用請求的函數(shù)
// uri中除了class/function之外的段也會(huì)被傳遞給調(diào)用的函數(shù)
call_user_func_array(array(&$CI, $method), array_slice($URI->rsegments, 2));
//輸出最終的內(nèi)容到瀏覽器
if ($EXT->_call_hook('display_override') === FALSE)
{
$OUT->_display();
}
//*BASEPATH/system/core/Loader.php
//看一個(gè)Loader類加載model的例子。這里只列出了部分代碼
public function model($model, $name = '', $db_conn = FALSE)
{
$CI =& get_instance();
if (isset($CI->$name))
{
show_error('The model name you are loading is the name of a resource that is already being used: '.$name);
}
$model = strtolower($model);
//依次根據(jù)model類的path進(jìn)行匹配,如果找到了就加載
foreach ($this->_ci_model_paths as $mod_path)
{
if ( ! file_exists($mod_path.'models/'.$path.$model.'.php'))
{
continue;
}
if ($db_conn !== FALSE AND ! class_exists('CI_DB'))
{
if ($db_conn === TRUE)
{
$db_conn = '';
}
$CI->load->database($db_conn, FALSE, TRUE);
}
if ( ! class_exists('CI_Model'))
{
load_class('Model', 'core');
}
require_once($mod_path.'models/'.$path.$model.'.php');
$model = ucfirst($model);
//這里依然將model對象注冊成控制器類的成員變量。Loader在加載其他資源的時(shí)候也會(huì)這么做
$CI->$name = new $model();
$this->_ci_models[] = $name;
return;
}
// couldn't find the model
show_error('Unable to locate the model you have specified: '.$model);
}
//*BASEPATH/system/core/Model.php
//__get()是一個(gè)魔術(shù)方法,當(dāng)讀取一個(gè)未定義的變量的值時(shí)就會(huì)被調(diào)用
//如下是Model基類對__get()函數(shù)的一個(gè)實(shí)現(xiàn),使得在Model類內(nèi),可以像直接在控制器類內(nèi)一樣(例如$this->var的方式)去讀取它的變量
function __get($key)
{
$CI =& get_instance();
return $CI->$key;
更多信息請查看IT技術(shù)專欄