thinkphp 框架自动加载原理_这下你应该理解ThinkPHP的Loader自动加载了

 2023-09-19 阅读 27 评论 0

摘要:想了很久终于要开始系列文章的编写了,期望是写出提升和面试都可以搞定的系列文章。 当你看到本文时,如果你发现咔咔没有编写到的面试热点问题或者技术难点,期待评论区指出,一起完善。前言目前再整理PHP进阶路线图,如有好的建议咔咔会第一时

想了很久终于要开始系列文章的编写了,期望是写出提升和面试都可以搞定的系列文章。 当你看到本文时,如果你发现咔咔没有编写到的面试热点问题或者技术难点,期待评论区指出,一起完善。

前言

目前再整理PHP进阶路线图,如有好的建议咔咔会第一时间进行收录。

一、自动加载loader源码分析

1-1 学习目标

  • 类的自动加载
  • 类自动加载的两种方式
  • spl_autoload_register必须会使用
  • 实现自定义文件的类的自动加载

1-2 Composer加载

c39a19e9faacab58f1f83dbd56d95cb4.png

从上图咔咔给的解析图,在base.php中首先加载了loader类,接着调用了register这个方法。

581fb14f05cf90eb7c6beadb96be05dc.png

来到thinkphplibraryhinkLoader.php有一个register的方法,在这个方法里边,我们先学习第一个知识点spl_autoload_register()聊聊spl_autoload_register前世今生和简单使用,直接点击即可查看。

thinkphp5教程、紧接着就是项目的根路径和composer的路径。

1c7ef4dc554ff2763ed06ecd4bb18101.png

从这里开始就是在加载composer文件,过程也是很简单

  • 1.判断composer是否为目录
  • 2.判断路径下面的autoload_static.php是否为文件
  • 3.引入autoload_static.php文件
  • 4.返回所有已经声明的所有类 数组返回
  • 5.获取最后一个类ComposerStaticInit30742487e00917c888d89ba216f165b9
  • 6.判断ComposerStaticInit30742487e00917c888d89ba216f165b9中是否存在数组中的数据
f725f6131ab0eb79301f0ed1430019a5.png

接着可以去vendorcomposerautoload_static.php文件中可以看到这俩个属性

eb514480b9416c04545ea3b3a28d3a1c.png

这里有一段代码估计有一部分同学会在这里绕一下self::${$attr} = $composerClass::${$attr};,这里的$attr就是'prefixLengthsPsr4', 'prefixDirsPsr4', 'fallbackDirsPsr4', 'prefixesPsr0', 'fallbackDirsPsr0', 'classMap', 'files'这些数据,外层在加一个$符号。

从而在ComposerStaticInit30742487e00917c888d89ba216f165b9这个类中直接获取对应的属性值,也就是上图的俩个属性值。

9113f8ee79c787f2de0be74014250adf.png

1-3 注册命名空间

php 自动加载是否安全,文件还是thinkphplibraryhinkLoader.php的register方法

在这里注册了俩个命令空间,分别为think和traits。然后会进入到addNamespace这个方法中

c7116633bb6a6418936766ebf2879f80.png

在addNamespace方法中,添加了Psr4空间

884a5b7359b6ca4d6a3dc3ec22b20aa4.png

接着来到addPsr4这个方法,会把这俩个命名空间都注册到ComposerStaticInit1e269472f484e157e90227b420ffca7a类的$prefixLengthsPsr4和$prefixDirsPsr4这俩个属性中

fe0b6ebaf9ac48fd46a93976369e5d84.png

为了验证上面做一个断点调试,看到这些数据就应该清晰了,至于traits也是一样的注册方式。

laravel thinkphp。截止到这里命名空间就注册完成了,接下来研究一下psr4命名空间是个什么东东。

54f093429cc8b7ca5299c674741d82a8.png

1-4 Psr4是什么玩意

psr是简单的理解就是文件路径、自动加载对应类的相关规范、目前TP5.1使用的是psr4规范

此处的类是指class、接口、超类结构

一个完整的类需要一下结构()*

以下规范来源于PHP文档

  • 完整的类名必须要有一个顶级命名空间,被称为 "vendor namespace";
  • 完整的类名可以有一个或多个子命名空间;
  • 完整的类名必须有一个最终的类名;
  • 完整的类名中任意一部分中的下滑线都是没有特殊含义的;
  • 完整的类名可以由任意大小写字母组成;
  • 所有类名都必须是大小写敏感的。

怎么做php框架。以下是官方给的一个例子,这个psr规范能理解就尽量去理解它

fc43686f3048246b613c0921f3d95209.png

1-5 加载类库映射文件

到这里,肯定会有一个疑问,这里怎么没有classmap.php这个文件。

da84cfa85887b210dcd229061a765f3a.png

最终会走到addClassMap这个方法,在这个方法中,只是把classmap.php这个文件的数据赋值给$classMap 而已,没有什么其它的用法

bc71063101e9cefcebed79bf8921af17.png

1-6 自动加载extend目录

extend这个目录用过TP框架的都多少用过的,在这个目录里边可以存放一下自定义的类库文件。

根据下图可以看到就是使用addAutoLoadDir这个方法进行加载的

1441fbb56139d97ff31a44a815ea6ab5.png

thinkphp路由原理,在方法中也仅仅是把extend的路径赋值给了$fallbackDirsPsr4这个属性。

6fb694ec4e11fc6823ae9ab15bad1698.png

截止到这里Loader::register();这部分就结束了,接着我们深入的看一下内部实现和实践案例。

在以上阅读源码中有四个属性,简单的整理一下

964368e997e331e43a87545512910167.png

二、简说类的加载过程

bde4f022310b0a415b03bc91f69aa528.png

在刚刚开始解析这里的源码时就有一个函数spl_autoload_register

当需要使用的类没有被引入时,这个函数会在PHP报错前被触发,未定义的类名会被当作参数传入这里会直接去执行thinkLoader::autoload这个方法

f817c7df99bcfe4a3b60885f390159f0.png

thinkphp6框架,经过断点第一个未加载的类就是thinkError

eff9ebcbbb37be9312adc251cdae399a.png

为什么是thinkError呢!可以在回到thinkphp/base.php看一下,当自动加载完执行完成后第一个执行的类就是Error

deb6ab3ac74f1cbcc12d8d2be71db785.png

可以简单的做个测试,将这Error改为Kaka,进行打印一下,这时的类就改变为Kaka。到这里大家对这个类的自动加载机制就有一定的了解了。

当使用的类没有被引入时会把这个类当做参数传到thinkphp/library/think/Loader.php的autoload方法中。

ae7f92e84cd409cdf678bb3fb5bf92c1.png

到这里在进行看一下autoload这个方法

cc2882d765780f516fb0f79d40857d48.png

php框架网站、先从findFile这个方法走,把未因为的类传入这个方法中,在findFile这个方法中会直接从classMap这个属性中直接把thinkError这个类映射的文件直接返回出来

f351641cb1a37666a26ba79a12f5f61e.png

将thinkError这个类的完整路径返回给autoload的file变量后,把win环境的大小写给判断了一次。

然后直接使用include引入文件即可,直到返回。

直到这里就是一次完整的类的自动加载解析。

0dbf25c99a5b964de3b611f1fed2d8eb.png

虽然到这里结束了,但是还是得再提一点就是$classMap这个属性,这个属性是基于文件classmap.php来到,这个文件的生成也是需要执行命令php think optimize:autoload生成的。

php 框架底层?当没有生成这个文件时程序是如何执行的呢!

之前的所有流程都是一样的,只有在findFile这里不一样,接下来进行简单的梳理一下。

这时代码肯定不会走classMap

6f2622e5a618d034bdf2c10e4eef6f49.png

先获取thinkError文件

f234361d363419bf7b5d1712308f0e28.png

然后经过Composer自动加载中的俩个属性进行获取命名空间,在把thinkError.php文件进行拼接

f29282672f8155277c8d278fb0c2f4b7.png

php自动加载,最终返回的结果也是D:phpstudy_proWWWThinkPHPSourceCodeAnalysishinkphplibraryhinkError.php这个文件。

这里的代码需要好好的阅读一下。

类的自动加载到这里就是完全结束了。

三、自定义文件如何实现类的自动加载

先创建一个文件夹kaka

2628e93d4ec66b68ee915de08cd8d5ee.png

这时在控制器index中引入文件Kaka.php

629b21869998cb197518134e87961231.png

tp5框架原理,直接进行访问,这时这个类肯定会报错,那么我们应该怎么操作一下,就可以直接访问呢!

bafc2174e5d88c52ef93f6c65ad6abec.png

这个时候就体现到源码的重要性了,还记得在自动加载的register函数中,加载过extend目录

fc242e27aa8d8f64c63b71580b006c8c.png

这时再加一个kaka这个目录,直接进行访问一下

2f762c26f781179979866172c59ec19c.png
5761528cae1a113e54b2fd5f15864171.png

没毛病,直接就出来了。一切OK

在这里再聊一下关于extent的加载方式

php框架路由实现原理?在之前聊注册自动加载类库目录只是说明了一下只是把路径存到了$fallbackDirsPsr4属性,没有细细说,接下来就是说明这些了。

阅读源码只能是实现那然后查看那

a34c42ffcde0c9d90372131a6c49407a.png

只要是定义的类都会进去到autoload进行自动加载

同样也会进入到findFile这个方法

f8852e43912f0f41f2ca0e1d8ca67f72.png

在findFile这个方法中可以看到这段代码,这个属性是不是很熟悉,就是自动加载extend目录时添加到$fallbackDirsPsr4属性的。

fef8dca8f9f6f1d8e1b7f980acc0b3c8.png

当在findFile中打印参数class时看一下数据

很清楚地可以看到testKaka这个类

12ad593083b589ab31ed7d99395b6409.png

此时再打印一下这个$fallbackDirsPsr4属性里边返回的file

3310760809f2d64dff46c26c1b327943.png

然后就是使用__include_file来直接includeD:phpstudy_proWWWThinkPHPSourceCodeAnalysiskakaestKaka.php我们定义的文件。

以上的这个自定义文件如何实现类的自动加载,并且也就是extend的加载方式

四、总结

关于类自动加载的所有流程就完成了,如有错误之处可以在评论区哦!

❝ 坚持学习、坚持写博、坚持分享是咔咔从业以来一直所秉持的信念。希望在偌大互联网中咔咔的文章能带给你一丝丝帮助。我是咔咔,下期见。

版权声明:本站所有资料均为网友推荐收集整理而来,仅供学习和研究交流使用。

原文链接:https://hbdhgg.com/4/77330.html

发表评论:

本站为非赢利网站,部分文章来源或改编自互联网及其他公众平台,主要目的在于分享信息,版权归原作者所有,内容仅供读者参考,如有侵权请联系我们删除!

Copyright © 2022 匯編語言學習筆記 Inc. 保留所有权利。

底部版权信息