1. 程式人生 > >Laravel 框架 自動載入實現分析

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 檔案。