Laravel 框架 自動載入實現分析
阿新 • • 發佈:2019-01-09
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); 會觸發致命錯誤,必須帶上名稱空間這是index.phpspl_autoload_register(array('app\Autoload','ss'),true); } public function ss(){ echo 666; exit; } }
<?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 檔案。