在laravel发现有些类可以直接use 类名,就能使用了,例如use DB;就可以使用DB类了,问题是DB这个类并不在根命名空间,这里面实际就是用到了别名。
先通过如下例子来分析基本原理
建立如下文件upload.php,内容为
<?php namespace test\test2; class upload{ public function test(){ return 123; } }
2 建立文件index.php,内容为
<?php namespace b; require('upload.php'); class_alias ( '\test\test2\upload' , 'upload'); $a=new \upload(); echo $a->test();
浏览器执行index.php,成功输出结果123;
可以看到class upload在命名空间test\test2下 但是new upload的时候 并没有new \test\test2\upload 而是直接new \upload,原因不多说,就是因为函数class_alias导致的.具体的这个函数的用法可以参考手册。这里要补充说明class_alias的第3个参数默认为true,手册上的意思是Whether to autoload if the original class is not found.是什么意思了,还是通过例子说明 ,把index.php修改如下
<?php namespace b; //require('upload.php'); spl_autoload_register(function($class){ $num=strrpos($class,'\\'); $num++; $file=substr($class, $num).'.php'; require($file); }); class_alias ( '\test\test2\upload' , 'upload'); $a=new \upload(); echo $a->test();
可以看到我注释掉了require('upload.php'),但是代码还是成功执行了。有了上面的例子说明,就能看懂laravel的别名实现机制了.
在laravel中,比方说我需要使用Log类,我们通过use Log; Log::info();就能使用记录日志了.下面来分析原理
laravel的加载过程这里不分析,中间有一步会执行如下这个'Illuminate\Foundation\Bootstrap\RegisterFacades'的bootstrap方法;
class RegisterFacades { public function bootstrap(Application $app) { //......省略...... AliasLoader::getInstance($app->make('config')->get('app.aliases'))->register(); } }
$app->make('config')->get('app.aliases')这一步读取了config文件夹下的app.php的配置文件,这个配置文件里面我们定义了别名列表.
意思就是说如果是需要使用别名 ,必须在配置文件中注册别名
例如配置文件中有一行配置为'Log' => Illuminate\Support\Facades\Log::class,
继续追踪执行流程,代码会执行到这一步
public function load($alias) { if (isset($this->aliases[$alias])) { return class_alias($this->aliases[$alias], $alias); } }
到了这里 ,看了之前的原理的人应该都明白了,我们new Log类 ,根据我们的配置'Log' => Illuminate\Support\Facades\Log::class,实际上是调用的Illuminate\Support\Facades\Log这个类, 可是Illuminate\Support\Facades\Log里并没有info方法,这是如何实现的了,可以百度facade原理,这里不细说,这里实际上调用的是是从容器里面获取到了log对象,那么这个log对象是什么时候注册到容器里面去的了,