array $vars = []){ try { // 通过反射实例化类 $reflect = new Reflect
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