zoukankan      html  css  js  c++  java
  • 【转】laravel aliases别名的原理

    在laravel发现有些类可以直接use 类名,就能使用了,例如use DB;就可以使用DB类了,问题是DB这个类并不在根命名空间,这里面实际就是用到了别名。

    先通过如下例子来分析基本原理

    建立如下文件upload.php,内容为

    <?php
    namespace test est2;
    class upload{
    public function test(){
    return 123;
    }
    }
    2 建立文件index.php,内容为

    <?php
    namespace b;
    require('upload.php');
    class_alias ( ' est est2upload' , 'upload');
    $a=new upload();
    echo $a->test();

    浏览器执行index.php,成功输出结果123;
    可以看到class upload在命名空间test est2下 但是new upload的时候 并没有new   est est2upload 而是直接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 ( ' est est2upload' , 'upload');
    $a=new upload();
    echo $a->test();

    可以看到我注释掉了require('upload.php'),但是代码还是成功执行了。有了上面的例子说明,就能看懂laravel的别名实现机制了.
    在laravel中,比方说我需要使用Log类,我们通过use Log; Log::info();就能使用记录日志了.下面来分析原理

    laravel的加载过程这里不分析,中间有一步会执行如下这个'IlluminateFoundationBootstrapRegisterFacades'的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' => IlluminateSupportFacadesLog::class,
    继续追踪执行流程,代码会执行到这一步

    public function load($alias)
    {
    if (isset($this->aliases[$alias])) {
    return class_alias($this->aliases[$alias], $alias);
    }
    }
    到了这里 ,看了之前的原理的人应该都明白了,我们new Log类 ,根据我们的配置'Log' => IlluminateSupportFacadesLog::class,实际上是调用的IlluminateSupportFacadesLog这个类,  可是IlluminateSupportFacadesLog里并没有info方法,这是如何实现的了,可以百度facade原理,这里不细说,这里实际上调用的是是从容器里面获取到了log对象,那么这个log对象是什么时候注册到容器里面去的了,

    protected $bootstrappers = [
    'IlluminateFoundationBootstrapDetectEnvironment',
    'IlluminateFoundationBootstrapLoadConfiguration',
    'IlluminateFoundationBootstrapConfigureLogging',
    'IlluminateFoundationBootstrapHandleExceptions',
    'IlluminateFoundationBootstrapRegisterFacades',
    'IlluminateFoundationBootstrapRegisterProviders',
    'IlluminateFoundationBootstrapBootProviders',
    ];

    这块代码,看过源码的人都应该熟悉了,在ConfigureLogging这个文件里就会将log注册到容器里面.这样一个完整的流程就结束了
    from:https://blog.csdn.net/jimgethelp/article/details/77869971

  • 相关阅读:
    string数组批量转换成Int数组
    TCP/IP 、 HTTP 、HTTPS
    静态布局、自适应布局、流式布局、响应式布局、弹性布局等的概念和区别
    Vue源码学习02 初始化模块init.js
    IOS8白屏
    VUE 源码学习01 源码入口
    http状态码
    vue全家桶(Vue+Vue-router+Vuex+axios)(Vue+webpack项目实战系列之二)
    Vue实战Vue-cli项目构建(Vue+webpack系列之一)
    module.exports,exports,export和export default,import与require区别与联系【原创】
  • 原文地址:https://www.cnblogs.com/xuan52rock/p/12442333.html
Copyright © 2011-2022 走看看