thinkphp6.0 底层源码分析 - 类的自动加载、配置文件初始化

本文主要是介绍thinkphp6.0 底层源码分析 - 类的自动加载、配置文件初始化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

概述

因为工作的需要,深入研究了一下thinkphp的源码,也算是对php知识的一个回归,工作这么多年,我一直坚信php是最好的Web编程语言,它可以做到成本和效率的一个平衡,知其然,更要知其所以然才是高手修炼之道

类的自动加载

不管是tp,yaf 还是yii ,所有的php框架都是从自动加载类库文件开始的,如果你不知道如何下手,就打开入口文件,从分析类的自动加载开始。

thinkphp6使用了composer去加载类库,整个composer的实现原理是:首先将各个使用了不同psr规范的类或映射类,以某种形式存储,然后当类找不到的时候,通过与存储的数据匹配,找到类所在的路径,然后去加载。

实际上composer总共有四种规范的文件需要加载,分别是:psr0、psr4、类映射、公共函数文件。

1.使用了单例模式,原理:简化后,psr0,psr4,classmap每个分类对应一个数组,类名在这三个数组进行检索,检索完成后,include

public static function getLoader()
{if (null !== self::$loader) {return self::$loader;}
}

2.此处先注册自动加载未定义类,紧跟着注销,是因为只加载并实例化classLoader类,其他类的加载,使用composer提供的方法,而不是自定义的。

spl_autoload_register(array('ComposerAutoloaderInit1283bda52466502421173f3a3bffb31b', 'loadClassLoader'), true, true);
self::$loader = $loader = new \Composer\Autoload\ClassLoader();
spl_autoload_unregister(array('ComposerAutoloaderInit1283bda52466502421173f3a3bffb31b', 'loadClassLoader'));

3.php版本大于5.6且未使用hhvm且没有启用zenGaurd加密扩展,即可使用静态加载,composer install 后,从各个vendor库的composer.json中读取autoload属性。

$useStaticLoader = PHP_VERSION_ID >= 50600 &&
!defined('HHVM_VERSION') &&
(!function_exists('zend_loader_file_encoded') ||
!zend_loader_file_encoded());

这里使用到了一个技巧,若对象类的成员属性是private,同时已经实现了set方法,现在需要实现同样的功能,直接复制给private成员属性。若是常用方法是将private属性变成public,或者修改set方法,或添加新的方法。但这里使用了系统类Closure的属性,可以通过bind方法,使用到了目标对象的private属性。

public static function getInitializer(ClassLoader $loader)
{return \Closure::bind(function () use ($loader) {$loader->prefixLengthsPsr4 = ComposerStaticInit1283bda52466502421173f3a3bffb31b::$prefixLengthsPsr4;$loader->prefixDirsPsr4 = ComposerStaticInit1283bda52466502421173f3a3bffb31b::$prefixDirsPsr4;$loader->fallbackDirsPsr0 = ComposerStaticInit1283bda52466502421173f3a3bffb31b::$fallbackDirsPsr0;$loader->classMap = ComposerStaticInit1283bda52466502421173f3a3bffb31b::$classMap;}, null, ClassLoader::class);
}

框架初始化执行流程

thinkphp6.0 应用的初始化做了大量的操作,其主要的操作有:加载环境变量、加载配置文件,加载语言包、监听 AppInit、initializers 数组包含的类的初始化。

public function run(Request $request = null): Response
{//初始化$this->initialize();//自动创建request对象$request = $request ?? $this->app->make('request', [], true);$this->app->instance('request', $request);try {$response = $this->runWithRequest($request);} catch (Throwable $e) {$this->reportException($e);$response = $this->renderException($request, $e);}return $response;
}

1.加载环境变量

重点强调一下在初始化加载initialize中,和底下的$this->lang->load$this->config->load都是一样的,都是加载对应文件中的数组。

// 加载环境变量
if (is_file($this->rootPath . '.env')) {$this->env->load($this->rootPath . '.env');
}object(think\Env)#8 (1) {["data":protected]=>array(14) {["APP_DEBUG"]=>string(4) "true"["DATABASE_CHARSET"]=>string(7) "utf8mb4"["PROJECT_WS_DOMAIN"]=>string(15) "wss://127.0.0.1"}
}

2.调试模式设置

t h i s − > d e b u g M o d e I n i t ( ) 运行原理详见注释 , 需要注意的是,这里不知道是不是源码中的 B u g , ! this->debugModeInit() 运行原理详见注释,需要注意的是,这里不知道是不是源码中的Bug,! this>debugModeInit()运行原理详见注释,需要注意的是,这里不知道是不是源码中的Bug!this->appDebug 恒为true (有时间在做ob缓存和依赖注入的知识点)。

protected function debugModeInit(): void
{// 应用调试模式if (!$this->appDebug) {$this->appDebug = $this->env->get('app_debug') ? true : false;// 关闭错误显示ini_set('display_errors', 'Off');}// 如果不是命令行模式if (!$this->runningInConsole()) {// 重新申请一块比较大的bufferif (ob_get_level() > 0) {// 相当于ob_get_contents() 和 ob_clean()// 获取缓冲区内容并删除缓冲区内容$output = ob_get_clean();}// 开启新的缓冲区控制ob_start();if (!empty($output)) {// 由于开启了新的缓冲区控制,// 这里不会直接输出$output// 而是等到依次执行了ob_flush()和flash()之后才将内容输出到浏览器echo $output;}}
}
  1. 加载应用文件和配置等操作

在加载全局初始化文件的时候,加载是有顺序的,首先加载app目录下的common.php文件和系统下的helper.php文件,然后加载config目录下的所有php文件,最后加载event事件和service服务文件。

protected function load(): void
{$appPath = $this->getAppPath();# 首先加载app目录下的common.php文件和系统下的helper.php文件if (is_file($appPath . 'common.php')) {include_once $appPath . 'common.php';}include_once $this->thinkPath . 'helper.php';$configPath = $this->getConfigPath();$files = [];if (is_dir($configPath)) {$files = glob($configPath . '*' . $this->configExt);}foreach ($files as $file) {$this->config->load($file, pathinfo($file, PATHINFO_FILENAME));}# 然后加载config目录下的所有php文件# 最后加载event事件和service服务文件if (is_file($appPath . 'event.php')) {$this->loadEvent(include $appPath . 'event.php');}if (is_file($appPath . 'service.php')) {$services = include $appPath . 'service.php';foreach ($services as $service) {$this->register($service);}}}

4.初始化错误和异常处理、注册系统服务和初始化系统服务

最后,初始化错误和异常处理、注册系统服务和初始化系统服务,这几行代码做了比较多的操作:分别实例化包含在里面的类,然后调用其init方法。initializers 数组的值如下:

 // 初始化
foreach ($this->initializers as $initializer) {$this->make($initializer)->init($this);
}
protected $initializers = [Error::class,RegisterService::class,BootService::class,
];

结尾

thinkphp6.0的类的自动加载和初始化就介绍到这里了,知其然,更要知其所以然才是高手修炼之道。

这篇关于thinkphp6.0 底层源码分析 - 类的自动加载、配置文件初始化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/336563

相关文章

Python中的Walrus运算符分析示例详解

《Python中的Walrus运算符分析示例详解》Python中的Walrus运算符(:=)是Python3.8引入的一个新特性,允许在表达式中同时赋值和返回值,它的核心作用是减少重复计算,提升代码简... 目录1. 在循环中避免重复计算2. 在条件判断中同时赋值变量3. 在列表推导式或字典推导式中简化逻辑

SpringBoot中配置文件的加载顺序解读

《SpringBoot中配置文件的加载顺序解读》:本文主要介绍SpringBoot中配置文件的加载顺序,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot配置文件的加载顺序1、命令⾏参数2、Java系统属性3、操作系统环境变量5、项目【外部】的ap

Spring Boot读取配置文件的五种方式小结

《SpringBoot读取配置文件的五种方式小结》SpringBoot提供了灵活多样的方式来读取配置文件,这篇文章为大家介绍了5种常见的读取方式,文中的示例代码简洁易懂,大家可以根据自己的需要进... 目录1. 配置文件位置与加载顺序2. 读取配置文件的方式汇总方式一:使用 @Value 注解读取配置方式二

Java程序进程起来了但是不打印日志的原因分析

《Java程序进程起来了但是不打印日志的原因分析》:本文主要介绍Java程序进程起来了但是不打印日志的原因分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java程序进程起来了但是不打印日志的原因1、日志配置问题2、日志文件权限问题3、日志文件路径问题4、程序

Java数组初始化的五种方式

《Java数组初始化的五种方式》数组是Java中最基础且常用的数据结构之一,其初始化方式多样且各具特点,本文详细讲解Java数组初始化的五种方式,分析其适用场景、优劣势对比及注意事项,帮助避免常见陷阱... 目录1. 静态初始化:简洁但固定代码示例核心特点适用场景注意事项2. 动态初始化:灵活但需手动管理代

Java 正则表达式URL 匹配与源码全解析

《Java正则表达式URL匹配与源码全解析》在Web应用开发中,我们经常需要对URL进行格式验证,今天我们结合Java的Pattern和Matcher类,深入理解正则表达式在实际应用中... 目录1.正则表达式分解:2. 添加域名匹配 (2)3. 添加路径和查询参数匹配 (3) 4. 最终优化版本5.设计思

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

IDEA自动生成注释模板的配置教程

《IDEA自动生成注释模板的配置教程》本文介绍了如何在IntelliJIDEA中配置类和方法的注释模板,包括自动生成项目名称、包名、日期和时间等内容,以及如何定制参数和返回值的注释格式,需要的朋友可以... 目录项目场景配置方法类注释模板定义类开头的注释步骤类注释效果方法注释模板定义方法开头的注释步骤方法注

QT进行CSV文件初始化与读写操作

《QT进行CSV文件初始化与读写操作》这篇文章主要为大家详细介绍了在QT环境中如何进行CSV文件的初始化、写入和读取操作,本文为大家整理了相关的操作的多种方法,希望对大家有所帮助... 目录前言一、CSV文件初始化二、CSV写入三、CSV读取四、QT 逐行读取csv文件五、Qt如何将数据保存成CSV文件前言

pytorch自动求梯度autograd的实现

《pytorch自动求梯度autograd的实现》autograd是一个自动微分引擎,它可以自动计算张量的梯度,本文主要介绍了pytorch自动求梯度autograd的实现,具有一定的参考价值,感兴趣... autograd是pytorch构建神经网络的核心。在 PyTorch 中,结合以下代码例子,当你