自古有云:当苹果官方出10的时候,我们就可以买得起5了。顺着这个真理举一反三,tp都出到6版本了,不学下5.1的源码怎么能行?
tp5.1入口文件为index.php,上面写着醒目的几个大字“加载基础文件”
ok,跳转到base.php (vscode可以按f12快速跳转)
base.php文件加载了Loader类,这是tp实现文件加载的类,并调用了该类的register()的注册自动加载方法,再接着跳转到loader.php文件。
Loader类开头定义了一些私有的静态变量,先跳过,直接阅读register方法
register方法中调用了spl_autoload_register的自动加载方法,并将Loader类的autoload方法传入,快速定位到autoload方法。
该方法传入一个类名参数,我们知道,当代码执行到类名,并且该类的类文件没有被引入时,会调用spl_autoload_register注册的回调函数,并将类名作为参数传入,这里在框架初始化的时候第一个被传入的是Error类(看上图base.php中,执行Loader后调用了Error方法)。
当框架刚初始化的时候Loader类中的变量肯定都没有值,所以跳过第一个条件判断,因为是等于号,必为true,先不阅读findFile方法,其中有一句代码为__include_file($file);我们可以猜到它将该类的类文件引入。回到register方法。
调用rootpath获取获取文件夹的绝对路径,并将composer文件的绝对路径赋值给私有变量composerPath注意:DIRECTORY_SEPARATOR为文件分割符,在win环境下为‘/’,在linux环境下为‘\’。
代码进入该函数的第一个循环,用composer安装tp为true
跳转到该文件,该类文件定义了php框架自动加载的规范/规则
回到Loader.php继续阅读
之后register调用了 addNamespace方法,传入了think指向其路径和traits指向其路径的数组,光标快速定位,该函数又调用了addPsr4函数,直接定位到addpsr4
仔细观察addPsr4内部的逻辑判断条件,!prefix很明显不满足,进入elseif分支,prefixDirsPsr4存储的是命名空间到绝对路径的映射,通过断点调试可知,框架刚初始化的时候并没有think\的建,所以执行该逻辑分支(其他不执行的分支忽略),将think指向其路径和traits指向其路径的数组 按psr4规范存入私有变量。至此addPsr4和addNamespace功能结束,返回register
当阅读到这句时,我们意外的发现在runtime下没有classmap.php该文件,该代码的注释为加载类库的映射文件,我们翻阅tp的手册
生成类库映射文件 · ThinkPHP5.1完全开发手册 · 看云
执行该命令行生成映射文件
我们随便拿一条出来分析
'think\\App' => 'D:/phpstudy_pro/WWW/tp5/thinkphp/library/' . '/think/App.php',
我们可以看到键名是我们在使用tp过程中常用的空间名,值就是绝对路径,将两者进行了映射
回到register函数,它调用了addClassMap方法,将classmap中的数组赋值给变量classMap
最后看register函数的最后一行代码,调用了addAutoLoadDir,将extend目录传入变量fallbackDirsPsr4。
至此自动加载完毕。
等等是不是少了什么?除了第一行外,其他代码就是给私有变量赋值呀,
我们再回头看第一行代码传入的回调函数autoload,刚才挖了一个坑为findFile,再看这个函数
之后的代码都是对在变量中 针对该类名进行类文件路径的查找。
总结
该函数的思想就是根据PSR4规范,组织 命名空间、类名和类文件绝对路径 这三者之间的映射关系,为方便autload回调函数快速定位到类文件。
注意
1.调用extend中的类的命名空间直接 \ 开头
2. 当需要建立一个和 extend一样的文件夹用来自定义类的时候,需要在register中仿照extend向变量中添加