Laravel 框架 自动加载实现分析

laravel 框架是通过composer 实现的自动加载。

是通过  下面的代码实现的。

require_once __DIR__ . '/composer' . '/autoload_real.php';

return ComposerAutoloaderInit7b20e4d61e2f88170fbbc44c70d38a1f::getLoader();

首先我们对spl_autoload_register和spl_autoload_unregister 这两个函数进行解释一下。

spl_autoload_register 自动注册 一个或多个 自动加载函数,这些函数一般在 实例化类的时候,自动运行。spl_autoload_unregister 恰恰相反。

贴上我实验的代码:


这是autoload.php

<?php
/**
 * Created by PhpStorm.
 * User: Administrator
 * Date: 2017/12/7
 * Time: 14:10
 */

namespace app;

class Autoload {

    public function __construct()
    {
        $this->autoload();
    }
    public function  autoload(){
        //  spl_autoload_register(array('Autoload','ss'),true); 会触发致命错误,必须带上命名空间
        spl_autoload_register(array('app\Autoload','ss'),true);
    }
    public function ss(){
        echo 666;
        exit;
    }


}
这是index.php

<?php
/**
 * Created by PhpStorm.
 * User: Administrator
 * Date: 2017/12/7
 * Time: 14:10
 */

require 'autoload.php';

$autoload=new \app\Autoload();

$b=new B();// 此时自动运行自动加载函数

echo 77;
exit;

找到getLoader 这个函数,并对其进行分析:

 public static function getLoader()
    {
        if (null !== self::$loader) {
            return self::$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 ($map as $namespace => $path) {
            $loader->set($namespace, $path);
        }

        $map = require __DIR__ . '/autoload_psr4.php';
        foreach ($map as $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 ($includeFiles as $fileIdentifier => $file) { composerRequire7b20e4d61e2f88170fbbc44c70d38a1f($fileIdentifier, $file); } return $loader; }}
/***** 包围的部分,主要对ClassLoader 中的

$prefixesPsr0   、$prefixDirsPsr4  、$classMap 等属性进行赋值。即加载一些配置好的文件,在后面进行加载或寻找文件时候,就是从加载的配置文件中寻找。寻找要加载的类主要通过register 函数来实现。然后分析register函数。

public function register($prepend = false)
{
    spl_autoload_register(array($this, 'loadClass'), true, $prepend);
}
发现实际将该类中loadClass 函数注册为自动加载函数。于是开始分析loadClass函数,最终是通过findFile进行类的寻找。

public function findFile($class)
{
/// 特别注意 参数$class 是根据命名空间生成的class名称,具体请参考命名空间特性。
 // work around for PHP 5.3.0 - 5.3.2 https://bugs.php.net/50731
    if ('\\' == $class[0]) {
        $class = substr($class, 1);
    }

    // class map lookup  首先从加载的classMap 中寻找
    if (isset($this->classMap[$class])) {
        return $this->classMap[$class];
    }
    if ($this->classMapAuthoritative) {
        return false;
    }
// 从刚才加载的配置文件中寻找文件。先按照 psr4 规则寻找,再按照psr0 寻找
// 两种规则的不同主要是对下划线的处理方式。
 $file = $this->findFileWithExtension($class, '.php');

    // Search for Hack files if we are running on HHVM
    if ($file === null && defined('HHVM_VERSION')) {
        $file = $this->findFileWithExtension($class, '.hh');
    }

    if ($file === null) {
        // Remember that this class does not exist.
        return $this->classMap[$class] = false;
    }

    return $file;
}
至此register函数分析完。我们接着分析getLoader函数剩余代码。

$includeFiles = require __DIR__ . '/autoload_files.php';
foreach ($includeFiles as $fileIdentifier => $file) {
    composerRequire7b20e4d61e2f88170fbbc44c70d38a1f($fileIdentifier, $file);
}

这段代码其实就是加载autoload_file.php 文件。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值