Laravel源碼解析之從入口開始

clipboard.png

前言

提高能力的方法並不是使用更多工具,而是解刨本身所使用的工具。今天咱們從Laravel啓動的第一步開始講起。php

入口文件

laravel是單入口框架,全部請求必將通過index.phplaravel

define('LARAVEL_START', microtime(true)); // 獲取啓動時間

使用composer是現代PHP的標誌web

require __DIR__.'/../vendor/autoload.php'; // 加載composer -> autoload.php

加載啓動文件redis

$app = require_once __DIR__.'/../bootstrap/app.php';

獲取$app是laravel啓動的關鍵,也能夠說$app是用於啓動laravel內核的鑰匙🔑。隨後就是加載內核,載入服務提供者、門面所映射的實體類,中間件,最後到接收http請求並返回結果。bootstrap

$kernel = $app->make(Illuminate\Contracts\Http\Kernel::class); // 加載核心類

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

$response->send();

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

看似短短的4行代碼,這則是laravel的優雅之處。咱們開始深層次解刨。api

bootstrap\app.php

這個啓動文件也能夠看做是一個服務提供者,不過他並無boot,register方法。由於入口文件直接加載他,全部這些不必的方法就不存在了。數組

做爲啓動文件,首頁則是加載框架全部必須的要要件,例如app

  • registerBaseBindings
  • registerBaseServiceProviders
  • registerCoreContainerAliases,

這其中包括了不少基礎性的方法和類,例如composer

  • db [\Illuminate\Database\DatabaseManager::class]
  • auth [\Illuminate\Auth\AuthManager::class, \Illuminate\Contracts\Auth\Factory::class]
  • log [\Illuminate\Log\LogManager::class, \Psr\Log\LoggerInterface::class]
  • queue [\Illuminate\Queue\QueueManager::class, \Illuminate\Contracts\Queue\Factory::class, \Illuminate\Contracts\Queue\Monitor::class]
  • redis [\Illuminate\Redis\RedisManager::class, \Illuminate\Contracts\Redis\Factory::class]
  • 等等 ...

而$app這個在服務提供者的核心變量則就是Application實例化所得,而你在服務提供者內使用的make,bind,singleton來自他的父類Container,都說容器是laravel的核心概念。這塊的概念後續咱們會詳細的講解。框架

$app = new Illuminate\Foundation\Application(
    realpath(__DIR__.'/../')
);

上面咱們已經得到$app的實例化了,如今經過$app來註冊核心類、異常類,並將$app返回給index.php

$app->singleton(
    Illuminate\Contracts\Http\Kernel::class,
    App\Http\Kernel::class
);

$app->singleton(
    Illuminate\Contracts\Console\Kernel::class,
    App\Console\Kernel::class
);

$app->singleton(
    Illuminate\Contracts\Debug\ExceptionHandler::class,
    App\Exceptions\Handler::class
);

App\Http\Kernel

核心類了全部的

  • 系統中間件
  • 羣組中間件
  • 路由中間件

固然你須要使用中間件也是在這個類中加載,是常常被使用的一個文件。

protected $middleware = [
            \Illuminate\Foundation\Http\Middleware\CheckForMaintenanceMode::class,
            \Illuminate\Foundation\Http\Middleware\ValidatePostSize::class,
            \App\Http\Middleware\TrimStrings::class,
            \Illuminate\Foundation\Http\Middleware\ConvertEmptyStringsToNull::class,
            \App\Http\Middleware\TrustProxies::class,
        ];
        
        /**
         * The application's route middleware groups.
         *
         * @var array
         */
        protected $middlewareGroups = [
            'web' => [
                \App\Http\Middleware\EncryptCookies::class,
                \Illuminate\Cookie\Middleware\AddQueuedCookiesToResponse::class,
                \Illuminate\Session\Middleware\StartSession::class,
                // \Illuminate\Session\Middleware\AuthenticateSession::class,
                \Illuminate\View\Middleware\ShareErrorsFromSession::class,
                \App\Http\Middleware\VerifyCsrfToken::class,
                \Illuminate\Routing\Middleware\SubstituteBindings::class,
            ],
            
            'api' => [
                'throttle:60,1',
                'bindings',
            ],
        ];

這個核心類繼承自他的父類Illuminate\Foundation\Http\Kernel::class,核心類作了不少事情,它會將全部的中間件所有存儲到一個指定的數組,方便內核調用及其餘類調用。

namespace App\Http;
    
use App\Api\Middleware\VerifyApiToken;
use Illuminate\Foundation\Http\Kernel as HttpKernel;
    
class Kernel extends HttpKernel

回到起點

Laravel的啓動經歷了很繁瑣的一個過程。這也是Laravel優雅的關鍵點。

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

$response->send();

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

將請求傳入則完成了整個laravel的啓動,至於結果的返回則有開發者自行經過控制器或其餘可訪問類返回。

致謝

感謝你看到這裏,本篇文章源碼解析靠我的理解。若有出入請拍磚。

但願本篇文章能夠幫到你。謝謝

相關文章
相關標籤/搜索