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

相关文章

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

基于Redis自动过期的流处理暂停机制

《基于Redis自动过期的流处理暂停机制》基于Redis自动过期的流处理暂停机制是一种高效、可靠且易于实现的解决方案,防止延时过大的数据影响实时处理自动恢复处理,以避免积压的数据影响实时性,下面就来详... 目录核心思路代码实现1. 初始化Redis连接和键前缀2. 接收数据时检查暂停状态3. 检测到延时过

MySQL的配置文件详解及实例代码

《MySQL的配置文件详解及实例代码》MySQL的配置文件是服务器运行的重要组成部分,用于设置服务器操作的各种参数,下面:本文主要介绍MySQL配置文件的相关资料,文中通过代码介绍的非常详细,需要... 目录前言一、配置文件结构1.[mysqld]2.[client]3.[mysql]4.[mysqldum

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

Redis分布式锁中Redission底层实现方式

《Redis分布式锁中Redission底层实现方式》Redission基于Redis原子操作和Lua脚本实现分布式锁,通过SETNX命令、看门狗续期、可重入机制及异常处理,确保锁的可靠性和一致性,是... 目录Redis分布式锁中Redission底层实现一、Redission分布式锁的基本使用二、Red

Spring Boot项目如何使用外部application.yml配置文件启动JAR包

《SpringBoot项目如何使用外部application.yml配置文件启动JAR包》文章介绍了SpringBoot项目通过指定外部application.yml配置文件启动JAR包的方法,包括... 目录Spring Boot项目中使用外部application.yml配置文件启动JAR包一、基本原理

SpringBoot加载profile全面解析

《SpringBoot加载profile全面解析》SpringBoot的Profile机制通过多配置文件和注解实现环境隔离,支持开发、测试、生产等不同环境的灵活配置切换,无需修改代码,关键点包括配置文... 目录题目详细答案什么是 Profile配置 Profile使用application-{profil

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、