作為一個(gè)框架的基礎(chǔ),自動(dòng)加載其實(shí)就起到一個(gè)運(yùn)輸線路的作用,再者,TP5.0已經(jīng)拋棄了單字母函數(shù)以及大部分輔助函數(shù)的運(yùn)用,所以,Loader里實(shí)現(xiàn)了應(yīng)用程序必要的一些功能(db、model、controller、action等)本章以Loader類為基礎(chǔ)分析,當(dāng)然也是自己得再學(xué)習(xí)。
寫在分析之前:我會(huì)按照我認(rèn)為的精簡(jiǎn)類內(nèi)必要闡述的核心方法闡述。
總覽:
/*
注冊(cè)自動(dòng)加載機(jī)制
@param $autoload 自動(dòng)加載自定義[pathclass::function]
用例:Loader::register('think\\Loader::autoload');
*/
function register($autoload){}
/*
自動(dòng)加載
@param $class 需要加載的類
用例:搭配register使用
*/
function autoload($class){} //自動(dòng)加載
/*
導(dǎo)入所需的類庫(kù)
@param string $class 類庫(kù)命名空間字符串
@param string $baseUrl 起始路徑*
@param string $ext 導(dǎo)入的文件擴(kuò)展名
用例:Loader::import('@.util.upload');
用例:Loader::import('qrcode', 'vendor');
用例:Loader::import('wechat-sdk.wechat', EXTEND_PATH, '.class.php');
*/
function import($class, $baseUrl, $ext){} //導(dǎo)入類庫(kù)
/*
實(shí)例化模型
@param $name Model名稱
@param $layer 業(yè)務(wù)層名稱
@param $appendSuffix 是否添加類名后綴 類似TP低版本的model業(yè)務(wù)層后綴或者其他業(yè)務(wù)層的后綴
@param $common 公共模塊名
用例:Loader::model('User');
*/
function model($name = '', $layer = 'model', $appendSuffix = false,$common = 'common'){} //
/*
實(shí)例化控制器
其他說(shuō)明同上
*/
function controller($name, $layer, $appendSuffix, $empty){}
/*
遠(yuǎn)程調(diào)controller的function
其他說(shuō)明同上
*/
function action($url, $vars, $layer, $appendSuffix){}
/*
實(shí)例化驗(yàn)證器
其他說(shuō)明同上
*/
function validate($name, $layer, $appendSuffix, $common){}
/*
實(shí)例化數(shù)據(jù)庫(kù)
@param $config 數(shù)據(jù)庫(kù)相關(guān)配置
用法:Loader::db();
*/
function db($config){} //實(shí)例化一個(gè)db類
class Loader{
protected static $map = []; //類名映射 ['class' =>'classpath',……]
protected static $namespaceAlias = []; //命名空間別名 ['path\namespace' =>''namespace_alias,……]
private static $autoloadFiles = []; //自動(dòng)加載文件
function register($class)
{
//注冊(cè)autoload function
spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true);
//添加tink命名空間
self::addNamespace([ 'think' => LIB_PATH . 'think' . DS]);
//添加類庫(kù)映射文件
self::addClassMap(__include_file(RUNTIME_PATH . 'classmap' . EXT));
//composer自動(dòng)加載支持
self::registerComposerLoader();
}
function autoload($class)
{
//加載命名空間別名
if (!empty(self::$namespaceAlias)) {
$namespace = dirname($class);
$original = self::$namespaceAlias[$namespace] . '\\' . basename($class);
//class_alias通過(guò)$original類(self::$namespaceAlias)創(chuàng)建一個(gè)別名$class
return class_alias($original, $class, false);
}
//加載類庫(kù)映射
if ($file = self::findFile($class)) {
include $file;
}
}
public static function import($class, $baseUrl = '', $ext = EXT)
{
static $_file = [];
$key = $class . $baseUrl;
if (empty($baseUrl)) {
list($name, $class) = explode(DS, $class, 2);
/*
如果已存在于psr4找到直接注冊(cè)命名空間,如果$class以@加載當(dāng)前模塊下,反之加載$class目錄
*/
if (isset(self::$prefixDirsPsr4[$name . '\\'])) {
$baseUrl = self::$prefixDirsPsr4[$name . '\\'];
} elseif ('@' == $name) {
$baseUrl = App::$modulePath;
}elseif (is_dir(EXTEND_PATH . $name)) {
$baseUrl = EXTEND_PATH;
}else{
$baseUrl = APP_PATH . $name . DS;
}
}elseif (substr($baseUrl, -1) != DS) {
$baseUrl .= DS;
}
//循環(huán)去找這個(gè)baseurl 發(fā)現(xiàn)直接返回找到的filename
if (is_array($baseUrl)) {
foreach ($baseUrl as $path) {
$filename = $path . DS . $class . $ext;
}
} else {
$filename = $baseUrl . $class . $ext;
}
__include_file($filename);
$_file[$key] = true;
}
public static function model($name = '', $layer = 'model', $appendSuffix = false, $common = 'common')
{
//單例一個(gè)model模型
if (isset(self::$instance[$name . $layer])) {
return self::$instance[$name . $layer];
}
……
//這里同controller一級(jí)action一樣解析module獲取模型類
$class = self::parseClass($module, $layer, $name, $appendSuffix);
if (class_exists($class)) {
$model = new $class();
} else {
/*
這里我精簡(jiǎn)了源碼邏輯,意思其實(shí)就是如果在model模型下沒(méi)找到這個(gè)模型類去找指定文件,
下定義的模型類(默認(rèn)common)所以我們公用模型便可以寫在這里,這點(diǎn)參考官方文檔。
找不到直接拋出 classnotfoundException 異常
*/
$model = new $module\$common\$class();
}
}
public static function controller($name, $layer = 'controller', $appendSuffix = false, $empty = '')
{
//$name[$module/$name] 如果不指定module則默認(rèn)當(dāng)前module
if (strpos($name, '/')) {
list($module, $name) = explode('/', $name);
} else {
$module = Request::instance()->module();
}
//解析組裝應(yīng)用類名
$class = self::parseClass($module, $layer, $name, $appendSuffix);
//如果類不存在,$empty為如果找不到這個(gè)控制器自定義的一個(gè)類;反則new $class
if (class_exists($class)) {
return new $class(Request::instance());
}elseif($empty && class_exists(self::parseClass($module, $layer, $empty , $appendSuffix))){
return new $emptyClass(Request::instance());
}
}
public static function action($url, $vars = [], $layer = 'controller', $appendSuffix = false)
{
//分拆$url[$module/$controller/$action]
$module = '.' != $info['dirname'] ? $info['dirname'] : Request::instance()->controller();
//實(shí)例化$url controller
$class = self::controller($module, $layer, $appendSuffix);
/*
封裝的php反射
$reflect = new \ReflectionMethod($class,$action);
$reflect->invokeArgs($class, $vars);
說(shuō)白了就是綁定類和類執(zhí)行方法,$vars為方法參數(shù)
詳細(xì)見(jiàn)源碼分析App類invokeMethod
*/
return App::invokeMethod([$class, $action . Config::get('action_suffix')], $vars);
}
public static function validate($name = '', $layer = 'validate', $appendSuffix = false, $common = 'common')
{
$name = $name ?: Config::get('default_validate');
//……
if (isset(self::$instance[$name . $layer])) {
return self::$instance[$name . $layer];
}
if (strpos($name, '/')) {
list($module, $name) = explode('/', $name);
} else {
$module = Request::instance()->module();
}
$class = self::parseClass($module, $layer, $name, $appendSuffix);
if (class_exists($class)) {
$validate = new $class;
} else {
//與model邏輯實(shí)現(xiàn)相似。都是找不到去找指定模塊下(默認(rèn)common)的驗(yàn)證類再找不到就classnotfound異常唄
$validate = new $module\$common\$class;
}
//單例存儲(chǔ)
self::$instance[$name . $layer] = $validate;
return $validate;
}
public static function db($config = [])
{
//實(shí)例化數(shù)據(jù)庫(kù) 詳細(xì)參Db類分析
return Db::connect($config);
}
}
在這里也貼一個(gè)qq群,用于交流Tp以及其他主流框架學(xué)習(xí):16585672