我獨自走進 Laravel5.5 的❤(一)

姚志博發表於2018-03-23

記錄這個經歷不是為了說我多厲害,多牛逼,而是為了能夠加深自己對所學知識的理解。如果可以幫助到別人學習這個框架的話,小生不勝榮幸。

關於 laravel 的教程很多,我也是通過大量閱讀別人寫的教程對 laravel 有了一個比較表面的認識。我也不打算把這個當做教程來寫,因為我沒寫過教程,不知道怎麼寫,而且我個人是比較隨意的,請原諒。

既然我是走進去的,就把我當做apache吧,我看到的第一個檔案應該是專案中 public 的 index.php 檔案。

1.

define('LARAVEL_START', microtime(true));
解讀:在執行時定義一個常量 LARAVEL_START 為當前 Unix 時間戳。

2.

require __DIR__.'/../vendor/autoload.php';
解讀:引入一個自動載入器

2.1進入到 autoload.php 中

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

return ComposerAutoloaderInit94ec56b468e10706582fb576e66d9831::getLoader();
解讀:從 autoload_real.php 中引入自載入入口 getLoader

public static function getLoader()
{
    //如果 $loader 非空,返回 $loader
    if (null !== self::$loader) {
        return self::$loader;
    }

    //spl_autoload_register:註冊指定的函式 loadClassLoader 作為__autoload的實現, throw:丟擲異常,prepend:新增到佇列之首
    spl_autoload_register(array('ComposerAutoloaderInit94ec56b468e10706582fb576e66d9831', 'loadClassLoader'), true, true);
    // 建立 \Composer\Autoload\ClassLoader() 物件,並物件化類屬性 $loader
    self::$loader = $loader = new \Composer\Autoload\ClassLoader();
    // 登出已註冊的指定的__autoload()函式loadClassLoader
    spl_autoload_unregister(array('ComposerAutoloaderInit94ec56b468e10706582fb576e66d9831', 'loadClassLoader'));

    // 判斷 php 版本是否大於5.6,且沒有定義 HHVM 版本(一個高效能 PHP 執行引擎,類似於 apache ),且 (不存在函式 zend_loader_file_encoded 或未執行 zend_loader_file_encoded )
    $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
    if ($useStaticLoader) {
        // 載入 autoload_static.php 的 ComposerStaticInit94ec56b468e10706582fb576e66d9831 類,一個靜態資訊檔案,包含了需要自動載入的files路徑、psr 規範資訊、laravel 框架類的路徑
        require_once __DIR__ . '/autoload_static.php';

        //執行 ComposerStaticInit94ec56b468e10706582fb576e66d9831 類中的 getInitializer 靜態函式。call_user_func:執行作為第一個引數的回撥函式,如果執行失敗返回 false,避免發生錯誤
        /**
         public static function getInitializer(ClassLoader $loader)
        {
        return \Closure::bind(function () use ($loader) {
        $loader->prefixLengthsPsr4 = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$prefixLengthsPsr4;
        $loader->prefixDirsPsr4 = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$prefixDirsPsr4;
        $loader->prefixesPsr0 = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$prefixesPsr0;
        $loader->classMap = ComposerStaticInit94ec56b468e10706582fb576e66d9831::$classMap;

        }, null, ClassLoader::class);
        }
         */
        // getInitializer 其實就是在返回 files 路徑、psr 規範資訊、laravel 框架類的路徑
        call_user_func(\Composer\Autoload\ComposerStaticInit94ec56b468e10706582fb576e66d9831::getInitializer($loader));
    } else {
        //載入 autoload_namespaces.php,返回符合 prs0 自載入外掛的名稱空間-路徑陣列
        $map = require __DIR__ . '/autoload_namespaces.php';
        foreach ($map as $namespace => $path) {
            //向 $loader 新增自載入外掛
            $loader->set($namespace, $path);
        }

        //載入 autoload_psr4.php,返回符合 prs4 自載入外掛的名稱空間-路徑陣列
        $map = require __DIR__ . '/autoload_psr4.php';
        foreach ($map as $namespace => $path) {
            //向 $loader 新增自載入外掛
            $loader->setPsr4($namespace, $path);
        }

        //載入 autoload_classmap.php,返回各種需自載入類的名稱空間-路徑陣列
        $classMap = require __DIR__ . '/autoload_classmap.php';
        if ($classMap) {
            //向 $loader 新增自載入類
            $loader->addClassMap($classMap);
        }
    }

    // 預載入 autoloader
    $loader->register(true);

    // 載入框架檔案
    if ($useStaticLoader) {
        $includeFiles = Composer\Autoload\ComposerStaticInit94ec56b468e10706582fb576e66d9831::$files;
    } else {
        $includeFiles = require __DIR__ . '/autoload_files.php';
    }
    foreach ($includeFiles as $fileIdentifier => $file) {
        composerRequire94ec56b468e10706582fb576e66d9831($fileIdentifier, $file);
    }

    return $loader;
}

解讀:檢視程式碼註釋

3.

$app = require_once __DIR__.'/../bootstrap/app.php';
解讀:引入laravel框架例項

4.


$kernel = $app->make(Illuminate\Contracts\Http\Kernel::class);

$response = $kernel->handle(
    $request = Illuminate\Http\Request::capture()
);

$response->send();

$kernel->terminate($request, $response);

解讀:啟動laravel應用與 http 互動,完成 http 請求和響應

/**
     * Resolve the given type from the container.
     * 對映指定型別的容器
     * (Overriding Container::make)
     *
     * @param  string  $abstract
     * @param  array  $parameters
     * @return mixed
     */
    public function make($abstract, array $parameters = [])
    {
        $abstract = $this->getAlias($abstract);

        if (isset($this->deferredServices[$abstract]) && ! isset($this->instances[$abstract])) {
            $this->loadDeferredProvider($abstract);
        }

        return parent::make($abstract, $parameters);
    }

// Illuminate\Container\Container::class
/**
     * Get the alias for an abstract if available.
     * 獲取可信賴的類組
     * @param  string  $abstract
     * @return string
     *
     * @throws \LogicException
     */
    public function getAlias($abstract)
    {
        if (! isset($this->aliases[$abstract])) {
            return $abstract;
        }

        if ($this->aliases[$abstract] === $abstract) {
            throw new LogicException("[{$abstract}] is aliased to itself.");
        }

        return $this->getAlias($this->aliases[$abstract]);
    }

    // Illuminate\Foundation\Application::class
    public function loadDeferredProvider($service)
    {
        if (! isset($this->deferredServices[$service])) {
            return;
        }

        $provider = $this->deferredServices[$service];

        // If the service provider has not already been loaded and registered we can
        // register it with the application and remove the service from this list
        // of deferred services, since it will already be loaded on subsequent.
        // 當服務提供者即將被載入,如果服務提供者還沒有被載入和註冊,
        // 我們可以在這個應用中註冊它,並且去除
        // 這個列表中已經快取的服務
        if (! isset($this->loadedProviders[$provider])) {
            $this->registerDeferredProvider($provider, $service);
        }
    }

    public function registerDeferredProvider($provider, $service = null)
    {
        // Once the provider that provides the deferred service has been registered we
        // will remove it from our local list of the deferred services with related
        // providers so that this container does not try to resolve it out again.
        // 一旦服務提供者提供被註冊的快取的服務,我們將從我們本地快取的服務列表中去除它和
        //與它相關聯的服務提供者,如此這個容器就不會再嘗試對映它。
        if ($service) {
            unset($this->deferredServices[$service]);
        }

        $this->register($instance = new $provider($this));

        if (! $this->booted) {
            $this->booting(function () use ($instance) {
                $this->bootProvider($instance);
            });
        }
    }

相關文章