Laravel框架中composer自动加载的实现分析
基础
自动加载允许你通过即用即加载的方式来加载需要的类文件,而不用每次都写繁琐的require和include语句。因此,每一次请求的执行过程都只加载必须的类,也不不要关心类的加载问题,只要需要的时候直接使用即可。
laravel框架是通过composer实现的自动加载。
是通过 下面的代码实现的。
require_once__DIR__.'/composer'.'/autoload_real.php'; returnComposerAutoloaderInit7b20e4d61e2f88170fbbc44c70d38a1f::getLoader();
首先我们对spl_autoload_register和spl_autoload_unregister这两个函数进行解释一下。
spl_autoload_register自动注册一个或多个自动加载函数,这些函数一般在实例化类的时候,自动运行。
spl_autoload_unregister恰恰相反。
贴上我实验的代码:
这是autoload.php
autoload(); } publicfunctionautoload(){ //spl_autoload_register(array('Autoload','ss'),true);会触发致命错误,必须带上命名空间 spl_autoload_register(array('app\Autoload','ss'),true); } publicfunctionss(){ echo666; exit; } }
这是index.php
找到getLoader这个函数,并对其进行分析:
publicstaticfunctiongetLoader() { if(null!==self::$loader){ returnself::$loader; } //注册自动加载函数,在加载或实例化类,运行loadClassLoader函数 spl_autoload_register(array('ComposerAutoloaderInit7b20e4d61e2f88170fbbc44c70d38a1f','loadClassLoader'),true,true); self::$loader=$loader=new\Composer\Autoload\ClassLoader(); spl_autoload_unregister(array('ComposerAutoloaderInit7b20e4d61e2f88170fbbc44c70d38a1f','loadClassLoader')); /********************1******************************************************** $map=require__DIR__.'/autoload_namespaces.php'; foreach($mapas$namespace=>$path){ $loader->set($namespace,$path); } $map=require__DIR__.'/autoload_psr4.php'; foreach($mapas$namespace=>$path){ $loader->setPsr4($namespace,$path); } $classMap=require__DIR__.'/autoload_classmap.php'; if($classMap){ $loader->addClassMap($classMap); } /********************1******************************************************** $loader->register(true);$includeFiles=require__DIR__.'/autoload_files.php';foreach($includeFilesas$fileIdentifier=>$file){composerRequire7b20e4d61e2f88170fbbc44c70d38a1f($fileIdentifier,$file);}return$loader;}}/*****包围的部分,主要对ClassLoader中的
$prefixesPsr0 、$prefixDirsPsr4 、$classMap等属性进行赋值。即加载一些配置好的文件,在后面进行加载或寻找文件时候,就是从加载的配置文件中寻找。寻找要加载的类主要通过register函数来实现。然后分析register函数。
publicfunctionregister($prepend=false) { spl_autoload_register(array($this,'loadClass'),true,$prepend); }发现实际将该类中loadClass函数注册为自动加载函数。于是开始分析loadClass函数,最终是通过findFile进行类的寻找。
publicfunctionfindFile($class) { ///特别注意参数$class是根据命名空间生成的class名称,具体请参考命名空间特性。 //workaroundforPHP5.3.0-5.3.2https://bugs.php.net/50731 if('\\'==$class[0]){ $class=substr($class,1); } //classmaplookup首先从加载的classMap中寻找 if(isset($this->classMap[$class])){ return$this->classMap[$class]; } if($this->classMapAuthoritative){ returnfalse; } //从刚才加载的配置文件中寻找文件。先按照psr4规则寻找,再按照psr0寻找 //两种规则的不同主要是对下划线的处理方式。 $file=$this->findFileWithExtension($class,'.php'); //SearchforHackfilesifwearerunningonHHVM if($file===null&&defined('HHVM_VERSION')){ $file=$this->findFileWithExtension($class,'.hh'); } if($file===null){ //Rememberthatthisclassdoesnotexist. return$this->classMap[$class]=false; } return$file; }至此register函数分析完。我们接着分析getLoader函数剩余代码。
$includeFiles=require__DIR__.'/autoload_files.php'; foreach($includeFilesas$fileIdentifier=>$file){ composerRequire7b20e4d61e2f88170fbbc44c70d38a1f($fileIdentifier,$file); }这段代码其实就是加载autoload_file.php文件。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对毛票票的支持。