位置:public/index.php
作用:
系统入口、常量定义、加载base.php、启动程序。
加载 thinkphp/base.php,并用Container类启动应用。
位置:thinkphp/base.php
thinkphp为php核心类库,5.1的核心类库不用composer加载,即thinkphp和vendor是同一文件夹下两个文件。
作用:
加载Loader.php,加载类和异常处理,实现日志接口,注册类库别名。
注册的别名大多都是think\facade中的类,包括App、Build、Cache、Config、Cookie、Db、Debug、Env、Facade、Hook、Lang、Log、Request、Response、Route、Session、Url、Validate、View。设置别名则代表创建think\App等类,在之后的代码中可以直接使用think\App。其他版本中若没有设置别名,则需使用think\facade\App等,这应该也是我用别的版本使用think\Log报错的原因。
位置:thinkphp/library/think/loader.php
作用:
主要功能就是加载类。
base.php 文件中有register()和addClassAlias()方法的使用。
该文件中根目录为public上层目录,若使用命令行则为脚本使用的目录。
register()首先调用autoload(),改方法若有参数则在对应数组中查找文件路径并直接加载文件。
之后自动加载根目录下/vendor/composer/autoload_static.php。
这段挺有意思
public static function register($autoload = '')
{
// 注册系统自动加载
spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true);
$rootPath = self::getRootPath();
self::$composerPath = $rootPath . 'vendor' . DIRECTORY_SEPARATOR . 'composer' . DIRECTORY_SEPARATOR;
// Composer自动加载支持
if (is_dir(self::$composerPath)) {
if (is_file(self::$composerPath . 'autoload_static.php')) {
require self::$composerPath . 'autoload_static.php';
$declaredClass = get_declared_classes();
$composerClass = array_pop($declaredClass);
foreach (['prefixLengthsPsr4', 'prefixDirsPsr4', 'fallbackDirsPsr4', 'prefixesPsr0', 'fallbackDirsPsr0', 'classMap', 'files'] as $attr) {
if (property_exists($composerClass, $attr)) {
self::${$attr} = $composerClass::${$attr};
}
}
} else {
self::registerComposerLoader(self::$composerPath);
}
}
// 注册命名空间定义
self::addNamespace([
'think' => __DIR__,
'traits' => dirname(__DIR__) . DIRECTORY_SEPARATOR . 'traits',
]);
// 加载类库映射文件
if (is_file($rootPath . 'runtime' . DIRECTORY_SEPARATOR . 'classmap.php')) {
self::addClassMap(__include_file($rootPath . 'runtime' . DIRECTORY_SEPARATOR . 'classmap.php'));
}
// 自动加载extend目录
self::addAutoLoadDir($rootPath . 'extend');
}
public static function registerComposerLoader($composerPath)
{
if (is_file($composerPath . 'autoload_namespaces.php')) {
$map = require $composerPath . 'autoload_namespaces.php';
foreach ($map as $namespace => $path) {
self::addPsr0($namespace, $path);
}
}
if (is_file($composerPath . 'autoload_psr4.php')) {
$map = require $composerPath . 'autoload_psr4.php';
foreach ($map as $namespace => $path) {
self::addPsr4($namespace, $path);
}
}
if (is_file($composerPath . 'autoload_classmap.php')) {
$classMap = require $composerPath . 'autoload_classmap.php';
if ($classMap) {
self::addClassMap($classMap);
}
}
if (is_file($composerPath . 'autoload_files.php')) {
self::$files = require $composerPath . 'autoload_files.php';
}
}
找到autoload_static.php文件之间返回路径,但之后还有代码估计是忘改了。
找不到autoload_static.php文件再找autoload_namespaces.php、autoload_psr4.php、autoload_classmap.php、autoload_files.php,找到后加到对应的数组里。
因为核心文件和vendor是分开的,找完这些文件还要加载核心文件,也是加到对应数组里。
再加载映射文件,为runtime/classmap.php文件,最后加载extend文件夹下的文件。
这些加载过程将项目中用到的文件的路径全部获取到,用于之后动态调用加载。
Loader::register()使用spl_autoload_register('think\\Loader::autoload'')将使用为定义的类时触发Loader::autoload()方法,实现动态加载类。
所以base.php里Error::register()先调用Loader::autoload()加载文件再调用Error::register()
总结以上内容动态加载类的方法为,先加载全部文件路径,再用spl_autoload_register()实现动态加载对应路径的文件的类。
位置:thinkphp/library/think/Error.php
作用:
Error::register()方法
public static function register()
{
error_reporting(E_ALL);
set_error_handler([__CLASS__, 'appError']);
set_exception_handler([__CLASS__, 'appException']);
register_shutdown_function([__CLASS__, 'appShutdown']);
}
error_reporting()设置报错级别,这里是全部。
set_error_handler() 函数设置用户自定义的错误处理函数,这里是Error::appError()。
set_exception_handler() 函数设置用户自定义的异常处理函数,Error::appException()。
register_shutdown_function()该函数是来注册一个会在PHP中止时执行的函数,Error::appShutdown()。