当前位置:首页 > Web开发 > 正文

array $vars = []){ try { // 通过反射实例化类 $reflect = new Reflect

2024-03-31 Web开发

ThinkPHP 6 从原先的 App 类中疏散出 Http 类,卖力应用的初始化和调理等成果,而 App 类则专注于容器的打点,切合单一职责原则。

以下源码分析,我们可以从 App,Http 类的实例化过程,了解类是如何实现自动实例化的,依赖注入是怎么实现的。

从入口文件出发

当访谒一个 ThinkPHP 搭建的站点,框架最先是从入口文件开始的,然后才是应用初始化、路由解析、控制器挪用和响应输出等操纵。

入口文件主要代码如下:

// 引入自动加载器,实现类的自动加载成果(PSR4标准) // 比拟Laravel、Yii2、Thinkphp的自动加载实现,它们根基就都一样 // 具体实现可参考我之前写的Laravel的自动加载实现: // @link: https://learnku.com/articles/20816 require __DIR__ . /../vendor/autoload.php; // 这一句和分为两部分分析,App的实例化和挪用「http」,具体见下文分析 $http = (new App())->http; $response = $http->run(); $response->send(); $http->end($response);

App 实例化

执行 new App() 实例化时,首先会挪用它的结构函数。

public function __construct(string $rootPath = ‘‘) { // thinkPath目录:如,D:\dev\tp6\vendor\topthink\framework\src\ $this->thinkPath = dirname(__DIR__) . DIRECTORY_SEPARATOR; // 项目根目录,如:D:\dev\tp6\ $this->rootPath = $rootPath ? rtrim($rootPath, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR : $this->getDefaultRootPath(); $this->appPath = $this->rootPath . app . DIRECTORY_SEPARATOR; $this->runtimePath = $this->rootPath . runtime . DIRECTORY_SEPARATOR; // 如果存在「绑定类库到容器」文件 if (is_file($this->appPath . provider.php)) { //将文件里的所有映射合并到容器的「$bind」成员变量中 $this->bind(include $this->appPath . provider.php); } //将当前容器实例生存到成员变量「$instance」中,也就是容器本身生存本身的一个实例 static::setInstance($this); // 生存绑定的实例到「$instances」数组中,见对应分析 $this->instance(app, $this); $this->instance(think\Container, $this); }

结构函数实现了项目各类根本路径的初始化,并读取了 provider.php 文件,将其类的绑定并入 $bind 成员变量,provider.php 文件默认内容如下:

return [ think\Request => Request::class, think\exception\Handle => ExceptionHandle::class, ];

合并后,$bind 成员变量的值如下:

$bind 的值是一组类的标识到类的映射。从这个实现也可以看出,我们不只可以在 provider.php 文件中添加标识到类的映射,而且可以笼罩其原有的映射,也就是将某些核心类替换本钱身界说的类。

static::setInstance($this) 实现的感化,如图:

think\App 类的 $instance 成员变量指向 think\App 类的一个实例,也就是类本身生存本身的一个实例。

instance() 要领的实现:

温馨提示: 本文由Jm博客推荐,转载请保留链接: https://www.jmwww.net/file/web/32896.html