浅谈 thinkphp composer 扩展包加载原理

本文将介绍 ThinkPHP 中 Composer 扩展包的加载原理,帮助读者更好地理解和应用该功能。


前言


如题,今天感觉好久没有更新博客了。最近迷上了物联网开发。一直在研究stm32、51这些东西。想起来前几天群里面有人问到tp扩展包原理。其实这个前几年也就研究过。网上搜了搜发现相关文章也很少(也有可能是我搜索姿势不对)今天就来写一篇thinkphp composer包加载原理


概览


  1. 当进行 composer update 或者 composer require 操作时。则会执行service:discover这个命令。把当前所有已经加载的库信息都进行一次匹配。如果匹配到了think关键字的services属性。则把服务类输出成配置文件到vendor/services.php文件中
  2. 当一次应用初始化(通常为一次访问开始时).则会引入vendor/services.php中的service服务类到当前应用内进行初始化

源码解析


composer包加载流程文字详解 建议先阅读一下这篇前两年我写的文章 Thinkphp6源码解析之分析 路由篇-请求流程

在第三步进入到Http->runWithRequest这个方法中后。可以看到又调用了initialize方法


image


追进这个方法可以看到


image


追进initialize方法看实现

    /**
     * 初始化应用
     * @access public
     * @return $this
     */
    public function initialize()
    {
        // 设置当前初始化状态
        $this->initialized = true;
        
        // 设置应用开始时间
        $this->beginTime = microtime(true);
        
        // 获取到php的内存
        $this->beginMem  = memory_get_usage();

        // 加载环境变量 例如当前应用目录下的 .env文件
        $this->loadEnv($this->envName);

        // 设置配置文件后缀
        $this->configExt = $this->env->get('config_ext', '.php');
        
        // 调试模式设置
        $this->debugModeInit();

        // 加载全局初始化文件
        $this->load();

        // 加载应用默认语言包
        $this->loadLangPack();

        // 监听AppInit
        $this->event->trigger(AppInit::class);
        
        // 设置php默认时区
        date_default_timezone_set($this->config->get('app.default_timezone', 'Asia/Shanghai'));

        // 初始化当前系统配置的默认服务
        foreach ($this->initializers as $initializer) {
			// 调用make函数生成对象。并且执行对象中的init方法
            $this->make($initializer)->init($this);
        }

        return $this;
    }

重点是初始化当前系统配置的默认服务这个$this->make($initializer)->init($this)函数,看看initializers属性


/**
     * 应用初始化器
     * @var array
     */
    protected $initializers = [
        Error::class,
        RegisterService::class,
        BootService::class,
    ];


追到这里就是关键了。上面把这里面的类进行初始化。并且执行类中的init方法。直接看RegisterService::class类的init方法

public function init(App $app)
    {
        // 获取当前项目根目录。拼接上 vendor/services.php
        $file = $app->getRootPath() . 'vendor/services.php';

        $services = $this->services;

        if (is_file($file)) {
            $services = array_merge($services, include $file);
        }

        // 初始化services
        foreach ($services as $service) {
            if (class_exists($service)) {
                $app->register($service);
            }
        }
    }

读到这里的可以看看自己项目vendor目录下是不是有一个services.php,接下来讲一讲composer.json这个文件 在tp框架中的composer.json有这样一个配置


image


这里这个概念我直接让chatgpt来解读。解读内容如下


image


接下来直接看service:discover这个命令。追到vendor\topthink\framework\src\think\console\command\ServiceDiscover.php文件


  public function execute(Input $input, Output $output)
    {
        // 获取到当前项目根目录下的 vendor/composer/installed.json 文件
        if (is_file($path = $this->app->getRootPath() . 'vendor/composer/installed.json')) {
            // json解析
            $packages = json_decode(@file_get_contents($path), true);
            // Compatibility with Composer 2.0
            if (isset($packages['packages'])) {
                $packages = $packages['packages'];
            }

            $services = [];
            foreach ($packages as $package) {
                // 判断当前包是否在extra字段里面声明了think关键字中的services属性。如果声明了就把services给装载到services变量内
                if (!empty($package['extra']['think']['services'])) {
                    $services = array_merge($services, (array) $package['extra']['think']['services']);
                }
            }

            $header = '// This file is automatically generated at:' . date('Y-m-d H:i:s') . PHP_EOL . 'declare (strict_types = 1);' . PHP_EOL;

            // 用var_export函数把services变量打印成可读性代码。并且写入到根目录vendor目录下的services
            $content = '<?php ' . PHP_EOL . $header . "return " . var_export($services, true) . ';';

            file_put_contents($this->app->getRootPath() . 'vendor/services.php', $content);

            $output->writeln('<info>Succeed!</info>');
        }

一直到这就算结束了


写在最后

如果觉得这篇文章对你有帮助。不妨点个赞留个关注再走

文章推荐

HTTP请求:requests的进阶使用方法浅析

创建nodejs项目并接入mysql,完成用户相关的增删改查的详细...

如何使用 Blazor 框架在前端浏览器中导入和导出 Excel

Java 20 新功能介绍

java获取到heapdump文件后,如何快速分析?

二叉树先序,中序,后序遍历的非递归算法(一)

和 chatgpt 聊了一会儿分布式锁 redis/zookeeper distribute...

RSA 简介及 C# 和 js 实现【加密知多少系列】

Cesium官方教程——Fabric

前端安全问题——暴破登录

读懂React原理之调和与Fiber

node使用multer进行文件上传