tp5源码解析--自动加载类

简介: 在TP5的框架使用过程中,自动加载类是可能会接触到,上手不难,但若想随心所欲的用,还是需要了解一番。用了千次,却没看过一次源码,学习源码,起码对TP5这个框架使用更加得心应手,毕竟技术服务于业务,能够写出更简介、更方便、更有效的业务代码,本身就是一件身心愉悦的事儿;

在TP5的框架使用过程中,自动加载类是可能会接触到,上手不难,但若想随心所欲的用,还是需要了解一番。用了千次,却没看过一次源码,学习源码,起码对TP5这个框架使用更加得心应手,毕竟技术服务于业务,能够写出更简介、更方便、更有效的业务代码,本身就是一件身心愉悦的事儿;

自动加载流程

第一步,TP框架初始化

Loader会走如下逻辑;

加载autoload_static.php中的 prefixLengthsPsr4prefixDirsPsr4两个数组;

public static $prefixLengthsPsr4 = array (
        't' => 
        array (
            'think\\composer\\' => 15,
            'think\\' => 6,
        ),
        'a' => 
        array (
            'app\\' => 4,
        ),
    );
    public static $prefixDirsPsr4 = array (
        'think\\composer\\' => 
        array (
            0 => __DIR__ . '/..' . '/topthink/think-installer/src',
        ),
        'think\\' => 
        array (
            0 => __DIR__ . '/../..' . '/thinkphp/library/think',
        ),
        'app\\' => 
        array (
            0 => __DIR__ . '/../..' . '/application',
        ),
    );


再通过self::${$attr} = $composerClass::${$attr} 变成当前类的静态数组;

详细逻辑如下:

// Composer 自动加载支持
        if (is_dir(VENDOR_PATH . 'composer')) {
            if (PHP_VERSION_ID >= 50600 && is_file(VENDOR_PATH . 'composer' . DS . 'autoload_static.php')) {
      //引入'autoload_static.php;
                require VENDOR_PATH . 'composer' . DS . 'autoload_static.php';
        //获取全部对象
                $declaredClass = get_declared_classes();
        //弹出最后一个对象类 也就是autoload_static.php
                $composerClass = array_pop($declaredClass);
    //查看autoload_static.php对象数组
                foreach (['prefixLengthsPsr4', 'prefixDirsPsr4', 'fallbackDirsPsr4', 'prefixesPsr0', 'fallbackDirsPsr0', 'classMap', 'files'] as $attr) {
                    if (property_exists($composerClass, $attr)) {
          //转化成自己类的静态数组
                        self::${$attr} = $composerClass::${$attr};
                    }
                }
            } else {
      //加载Psr0方法
                self::registerComposerLoader();
            }
        }
        // 注册命名空间定义
        self::addNamespace([
            'think'    => LIB_PATH . 'think' . DS,
            'behavior' => LIB_PATH . 'behavior' . DS,
            'traits'   => LIB_PATH . 'traits' . DS
        ]);
        // 加载类库映射文件
        if (is_file(RUNTIME_PATH . 'classmap' . EXT)) {
            self::addClassMap(__include_file(RUNTIME_PATH . 'classmap' . EXT));
        }
        self::loadComposerAutoloadFiles();
        // 自动加载 extend 目录,可以在这里加相关目录,拓展包就是在这手动放入 extend中加载
        self::$fallbackDirsPsr4[] = rtrim(EXTEND_PATH, DS);

第二步 new class()

当一个class不存在走自动加载方法; spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true) 通过spl_autoload_register 走autoload($class)方法;

autoload($class)->检测命名空间别名->通过上一步初始化生成的静态化数组查找相关class文件是否存在->返回对像(结束);

public static function autoload($class)
    {
        // 检测命名空间别名
        if (!empty(self::$namespaceAlias)) {
            $namespace = dirname($class);
            if (isset(self::$namespaceAlias[$namespace])) {
                $original = self::$namespaceAlias[$namespace] . '\\' . basename($class);
                if (class_exists($original)) {
                    return class_alias($original, $class, false);
                }
            }
        }
//查看文件
        if ($file = self::findFile($class)) {
            // 非 Win 环境不严格区分大小写
            if (!IS_WIN || pathinfo($file, PATHINFO_FILENAME) == pathinfo(realpath($file), PATHINFO_FILENAME)) {
      //引入文件
                __include_file($file);
                return true;
            }
        }
        return false;
    }


其中核心方法

public static function register($autoload = null)
    {
     // 注册系统自动加载
        spl_autoload_register($autoload ?: 'think\\Loader::autoload', true, true);
  }


TP自动加载方法

composer自动方法

其实两个自动加载打通小异;

Tp在加载方法,个人认为是借鉴composer;

目录
相关文章
|
9天前
|
安全 Java
并发编程之常见线程安全类以及一些示例的详细解析
并发编程之常见线程安全类以及一些示例的详细解析
13 0
|
2天前
|
缓存 Java 开发者
10个点介绍SpringBoot3工作流程与核心组件源码解析
Spring Boot 是Java开发中100%会使用到的框架,开发者不仅要熟练使用,对其中的核心源码也要了解,正所谓知其然知其所以然,V 哥建议小伙伴们在学习的过程中,一定要去研读一下源码,这有助于你在开发中游刃有余。欢迎一起交流学习心得,一起成长。
|
3天前
|
消息中间件 缓存 前端开发
Netty消息编码及发送源码解析
Netty消息编码及发送源码解析
6 0
|
6天前
|
XML 人工智能 Java
Spring Bean名称生成规则(含源码解析、自定义Spring Bean名称方式)
Spring Bean名称生成规则(含源码解析、自定义Spring Bean名称方式)
|
13天前
|
数据安全/隐私保护 C++
C++ 类方法解析:内外定义、参数、访问控制与静态方法详解
C++ 中的类方法(成员函数)分为类内定义和类外定义,用于操作类数据。类内定义直接在类中声明和定义,而类外定义则先在类中声明,再外部定义。方法可以有参数,访问权限可通过 public、private 和 protected 控制。静态方法与类关联,不依赖对象实例,直接用类名调用。了解这些概念有助于面向对象编程。
14 0
|
14天前
yolo-world 源码解析(六)(2)
yolo-world 源码解析(六)
45 0
|
14天前
yolo-world 源码解析(六)(1)
yolo-world 源码解析(六)
44 0
|
15天前
yolo-world 源码解析(五)(4)
yolo-world 源码解析(五)
47 0
|
15天前
yolo-world 源码解析(五)(1)
yolo-world 源码解析(五)
61 0
|
15天前
yolo-world 源码解析(二)(2)
yolo-world 源码解析(二)
58 0

推荐镜像

更多