[TOC]
* * * * *
## 1 自動加載器源代碼(thinkphp/library/think/Loader.php的加載器部分,其余的實例化部分分析見[另:自動加載器實例化](http://www.hmoore.net/zmwtp/tp5/120008))
~~~
protected static $map = [];
protected static $load = [];
protected static $namespace = [];
private static $prefixLengthsPsr4 = [];
private static $prefixDirsPsr4 = [];
private static $prefixesPsr0 = [];
~~~
~~~
public static function autoload($class)
{
if (isset(self::$map[$class])) {
if (is_file(self::$map[$class])) {
APP_DEBUG && self::$load[] = self::$map[$class];
include self::$map[$class];
}
} elseif ($file = self::findFileInComposer($class)) {
APP_DEBUG && self::$load[] = $file;
include $file;
} else {
if (!strpos($class, '\\')) {
return;
}
list($name, $class) = explode('\\', $class, 2);
if (isset(self::$namespace[$name])) {
$path = self::$namespace[$name];
} elseif (is_dir(EXTEND_PATH . $name)) {
$path = EXTEND_PATH . $name . DS;
} else {
return;
}
$filename = $path . str_replace('\\', DS, $class) . EXT;
if (is_file($filename)) {
if (APP_DEBUG && IS_WIN && false === strpos(realpath($filename), $class . EXT)) {
return;
}
APP_DEBUG && self::$load[] = $filename;
include $filename;
} else {
Log::record('autoloader error : ' . $filename, 'notice');
}
}
}
~~~
~~~
public static function addMap($class, $map = '')
{
if (is_array($class)) {
self::$map = array_merge(self::$map, $class);
} else {
self::$map[$class] = $map;
}
}
~~~
~~~
public static function addNamespace($namespace, $path = '')
{
if (is_array($namespace)) {
self::$namespace = array_merge(self::$namespace, $namespace);
} else {
self::$namespace[$namespace] = $path;
}
}
~~~
~~~
public static function register($autoload = '')
{
spl_autoload_register($autoload ? $autoload : 'think\\Loader::autoload');
self::registerComposerLoader();
}
private static function registerComposerLoader()
{
if (is_file(VENDOR_PATH . 'composer/autoload_namespaces.php')) {
$map = require VENDOR_PATH . 'composer/autoload_namespaces.php';
foreach ($map as $namespace => $path) {
self::$prefixesPsr0[$namespace[0]][$namespace] = (array) $path;
}
}
if (is_file(VENDOR_PATH . 'composer/autoload_psr4.php')) {
$map = require VENDOR_PATH . 'composer/autoload_psr4.php';
foreach ($map as $namespace => $path) {
$length = strlen($namespace);
if ('\\' !== $namespace[$length - 1]) {
throw new \InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");
}
self::$prefixLengthsPsr4[$namespace[0]][$namespace] = $length;
self::$prefixDirsPsr4[$namespace] = (array) $path;
}
}
if (is_file(VENDOR_PATH . 'composer/autoload_classmap.php')) {
$classMap = require VENDOR_PATH . 'composer/autoload_classmap.php';
if ($classMap) {
self::addMap($classMap);
}
}
if (is_file(VENDOR_PATH . 'composer/autoload_files.php')) {
$includeFiles = require VENDOR_PATH . 'composer/autoload_files.php';
foreach ($includeFiles as $fileIdentifier => $file) {
self::composerRequire($fileIdentifier, $file);
}
}
}
~~~
~~~
private static function composerRequire($fileIdentifier, $file)
{
if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) {
require $file;
$GLOBALS['__composer_autoload_files'][$fileIdentifier] = true;
}
}
~~~
~~~
private static function findFileInComposer($class, $ext = '.php')
{
$logicalPathPsr4 = strtr($class, '\\', DS) . $ext;
$first = $class[0];
if (isset(self::$prefixLengthsPsr4[$first])) {
foreach (self::$prefixLengthsPsr4[$first] as $prefix => $length) {
if (0 === strpos($class, $prefix)) {
foreach (self::$prefixDirsPsr4[$prefix] as $dir) {
if (file_exists($file = $dir . DS . substr($logicalPathPsr4, $length))) {
return $file;
}
}
}
}
}
if (false !== $pos = strrpos($class, '\\')) {
$logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1)
. strtr(substr($logicalPathPsr4, $pos + 1), '_', DS);
} else {
$logicalPathPsr0 = strtr($class, '_', DS) . $ext;
}
if (isset(self::$prefixesPsr0[$first])) {
foreach (self::$prefixesPsr0[$first] as $prefix => $dirs) {
if (0 === strpos($class, $prefix)) {
foreach ($dirs as $dir) {
if (file_exists($file = $dir . DS . $logicalPathPsr0)) {
return $file;
}
}
}
}
}
return self::$map[$class] = false;
}
~~~
~~~
public static function import($class, $baseUrl = '', $ext = EXT)
{
static $_file = [];
$class = str_replace(['.', '#'], [DS, '.'], $class);
if (isset($_file[$class . $baseUrl])) {
return true;
} else {
$_file[$class . $baseUrl] = true;
}
if (empty($baseUrl)) {
list($name, $class) = explode(DS, $class, 2);
if (isset(self::$namespace[$name])) {
$baseUrl = self::$namespace[$name];
} elseif ('@' == $name || MODULE_NAME == $name) {
$baseUrl = MODULE_PATH;
} elseif (in_array($name, ['traits', 'think', 'behavior'])) {
$baseUrl = LIB_PATH;
} elseif (APP_NAMESPACE == $name) {
$baseUrl = APP_PATH;
} elseif (is_dir(EXTEND_PATH . $name)) {
$baseUrl = EXTEND_PATH;
} else {
$baseUrl = APP_PATH . $name . DS;
}
} elseif (substr($baseUrl, -1) != DS) {
$baseUrl .= DS;
}
$filename = $baseUrl . $class . $ext;
if (is_file($filename)) {
if (APP_DEBUG && IS_WIN && false === strpos(realpath($filename), $class . $ext)) {
return false;
}
include $filename;
return true;
}
return false;
}
~~~
## 2 分析
Loader.php是框架的類的自動加載實現文件。可以使用autoload()自動擋加載類,import()手動加載指定目錄的類。
`protected static $map = [];`
系統的類名映射緩存數組,在下面的addMap()使用
` protected static $load = [];`
系統的加載列表緩存數組,在下面的autoload()使用
`protected static $namespace = [];`
系統的命名空間緩存數組,在下面的addNamespace()使用
~~~
private static $prefixLengthsPsr4 = [];
private static $prefixDirsPsr4 = [];
~~~
系統的PSR-4緩存數組,在下面的registerComposerLoader()使用
`private static $prefixesPsr0 = [];`
系統的PSR-0緩存數組,在下面的registerComposerLoader()使用。
上面四個靜態變量用作加載器的緩存數組,保證加載效率。
* * * * *
1 autoload($class)
根據類名自動加載。
`public static function autoload($class){}`
使用類庫映射加載對應的類。
~~~
if (isset(self::$map[$class])) {
if (is_file(self::$map[$class])) {
APP_DEBUG && self::$load[] = self::$map[$class];
include self::$map[$class];
}
}
~~~
使用composer加載類
~~~
elseif ($file = self::findFileInComposer($class)) {
APP_DEBUG && self::$load[] = $file;
include $file;
}
~~~
根據命名空間自動加載類
~~~
else {
if (!strpos($class, '\\')) {
return;
}
list($name, $class) = explode('\\', $class, 2);
if (isset(self::$namespace[$name])) {
$path = self::$namespace[$name];
} elseif (is_dir(EXTEND_PATH . $name)) {
$path = EXTEND_PATH . $name . DS;
} else {
return;
}
$filename = $path . str_replace('\\', DS, $class) . EXT;
if (is_file($filename)) {
if (APP_DEBUG && IS_WIN && false === strpos(realpath($filename), $class . EXT)) {
return;
}
APP_DEBUG && self::$load[] = $filename;
include $filename;
} else {
Log::record('autoloader error : ' . $filename, 'notice');
}
}
~~~
* * * * *
2 addMap($class,$map='')
注冊類名映射
`public static function addMap($class, $map = ''){}`
數組合并到$map,字符串關聯到$class.
~~~
if (is_array($class)) {
self::$map = array_merge(self::$map, $class);
} else {
self::$map[$class] = $map;
}
~~~
* * * * *
3 addNamespace($namespace,$path)
注冊命名空間和路徑的關聯
`public static function addNamespace($namespace, $path = ''){}`
數組合并到$namespace,字符串關聯到path.
~~~
if (is_array($namespace)) {
self::$namespace = array_merge(self::$namespace, $namespace);
} else {
self::$namespace[$namespace] = $path;
}
~~~
* * * * *
4 register($autoload='')
注冊自動加載機制
`public static function register($autoload = ''){}`
think\Loader::autoload()系統自動加載
self::registerComposerLoader() composer自動加載
~~~
spl_autoload_register($autoload ? $autoload : 'think\\Loader::autoload');
self::registerComposerLoader();
~~~
5 composer自動加載的注冊
`private static function registerComposerLoader(){}`
關于composer自動加載機制見 基礎原理的php的composer自動加載
* * * * *
6 composer自動加載的兩個私有方法
~~~
private static function composerRequire($fileIdentifier, $file){}
private static function findFileInComposer($class, $ext = '.php'){}
~~~
* * * * *
7 import($class,$baseUrl='',$ext=EXT)
手動加載$baseUrl下的$class命名的$ext后綴文件
`public static function import($class, $baseUrl = '', $ext = EXT){}`
首先分析$baseUrl,
然后查找$baserUrl.$class.$ext 文件名,然后加載對應文件
## 3 總結
Loader類是系統的系統的自動加載類。
五個靜態變量用來緩存自動加載的結果。
addMap() addNamespace() 用來添加$map,$namespace的關聯信息。
register() 注冊自動加載機制 autoload() 自動加載類
import()手動加載類
- 更新記錄
- 概述
- 文件索引
- 函數索引
- 章節格式
- 框架流程
- 前:章節說明
- 主:(index.php)入口
- 主:(start.php)框架引導
- 主:(App.php)應用啟動
- 主:(App.php)應用調度
- C:(Controller.php)應用控制器
- M:(Model.php)數據模型
- V:(View.php)視圖對象
- 附:(App.php)應用啟動
- 附:(base.php)全局變量
- 附:(common.php)模式配置
- 附:(convention.php)全局配置
- 附:(Loader.php)自動加載器
- 附:(Build.php)自動生成
- 附:(Hook.php)監聽回調
- 附:(Route.php)全局路由
- 附:(Response.php)數據輸出
- 附:(Log.php)日志記錄
- 附:(Exception.php)異常處理
- 框架工具
- 另:(helper.php)輔助函數
- 另:(Cache.php)數據緩存
- 另:(Cookie.php)cookie操作
- 另:(Console.php)控制臺
- 另:(Debug.php)開發調試
- 另:(Error.php)錯誤處理
- 另:(Url.php)Url操作文件
- 另:(Loader.php)加載器實例化
- 另:(Input.php)數據輸入
- 另:(Lang.php)語言包管理
- 另:(ORM.php)ORM基類
- 另:(Process.php)進程管理
- 另:(Session.php)session操作
- 另:(Template.php)模板解析
- 框架驅動
- D:(\config)配置解析
- D:(\controller)控制器擴展
- D:(\model)模型擴展
- D:(\db)數據庫驅動
- D:(\view)模板解析
- D:(\template)模板標簽庫
- D:(\session)session驅動
- D:(\cache)緩存驅動
- D:(\console)控制臺
- D:(\process)進程擴展
- T:(\traits)Trait目錄
- D:(\exception)異常實現
- D:(\log)日志驅動
- 使用范例
- 服務器與框架的安裝
- 控制器操作
- 數據模型操作
- 視圖渲染控制
- MVC開發初探
- 模塊開發
- 入口文件定義全局變量
- 運行模式開發
- 框架配置
- 自動生成應用
- 事件與插件注冊
- 路由規則注冊
- 輸出控制
- 多種應用組織
- 綜合應用
- tp框架整合后臺auto架構快速開發
- 基礎原理
- php默認全局變量
- php的魔術方法
- php命名空間
- php的自動加載
- php的composer
- php的反射
- php的trait機制
- php設計模式
- php的系統時區
- php的異常錯誤
- php的輸出控制
- php的正則表達式
- php的閉包函數
- php的會話控制
- php的接口
- php的PDO
- php的字符串操作
- php的curl
- 框架心得
- 心:整體結構
- 心:配置詳解
- 心:加載器詳解
- 心:輸入輸出詳解
- 心:url路由詳解
- 心:模板詳解
- 心:模型詳解
- 心:日志詳解
- 心:緩存詳解
- 心:控制臺詳解
- 框架更新
- 4.20(驗證類,助手函數)
- 4.27(新模型Model功能)
- 5.4(新數據庫驅動)
- 7.28(自動加載)