Laravel源码解析之从入口开始

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Laravel源码解析之从入口开始相关的知识,希望对你有一定的参考价值。

前言

提升能力的方法并非使用更多工具,而是解刨自己所使用的工具。今天我们从Laravel启动的第一步开始讲起。

入口文件

laravel是单入口框架,所有请求必将经过index.php


define('LARAVEL_START', microtime(true)); // 获取启动时间

使用composer是现代PHP的标志


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

加载启动文件


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

获取$app是laravel启动的关键,也可以说$app是用于启动laravel内核的钥匙??。随后就是加载内核,载入服务提供者、门面所映射的实体类,中间件,最后到接收http请求并返回结果。


$kernel = $app->make(IlluminateContractsHttpKernel::class); // 加载核心类

$response = $kernel->handle(
    $request = IlluminateHttpRequest::capture()
);

$response->send();

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

看似短短的4行代码,这则是laravel的优雅之处。我们开始深层次解刨。

bootstrapapp.php

这个启动文件也可以看作是一个服务提供者,不过他并没有boot,register方法。因为入口文件直接加载他,所有这些没必要的方法就不存在了。

作为启动文件,首页则是加载框架所有必须的要要件,例如

  • registerBaseBindings
  • registerBaseServiceProviders
  • registerCoreContainerAliases,

这其中包括了很多基础性的方法和类,例如

  • db [IlluminateDatabaseDatabaseManager::class]
  • auth [IlluminateAuthAuthManager::class, IlluminateContractsAuthFactory::class]
  • log [IlluminateLogLogManager::class, PsrLogLoggerInterface::class]
  • queue [IlluminateQueueQueueManager::class, IlluminateContractsQueueFactory::class, IlluminateContractsQueueMonitor::class]
  • redis [IlluminateRedisRedisManager::class, IlluminateContractsRedisFactory::class]
  • 等等 ...

而$app这个在服务提供者的核心变量则就是Application实例化所得,而你在服务提供者内使用的make,bind,singleton来自他的父类Container,都说容器是laravel的核心概念。这块的概念后续我们会详细的讲解。


$app = new IlluminateFoundationApplication(
    realpath(__DIR__.'/../')
);

上面我们已经获得$app的实例化了,现在通过$app来注册核心类、异常类,并将$app返回给index.php


$app->singleton(
    IlluminateContractsHttpKernel::class,
    AppHttpKernel::class
);

$app->singleton(
    IlluminateContractsConsoleKernel::class,
    AppConsoleKernel::class
);

$app->singleton(
    IlluminateContractsDebugExceptionHandler::class,
    AppExceptionsHandler::class
);

AppHttpKernel

核心类了所有的

  • 系统中间件
  • 群组中间件
  • 路由中间件

当然你需要使用中间件也是在这个类中加载,是经常被使用的一个文件。


protected $middleware = [
            IlluminateFoundationHttpMiddlewareCheckForMaintenanceMode::class,
            IlluminateFoundationHttpMiddlewareValidatePostSize::class,
            AppHttpMiddlewareTrimStrings::class,
            IlluminateFoundationHttpMiddlewareConvertEmptyStringsToNull::class,
            AppHttpMiddlewareTrustProxies::class,
        ];
        
        /**
         * The application's route middleware groups.
         *
         * @var array
         */
        protected $middlewareGroups = [
            'web' => [
                AppHttpMiddlewareEncryptCookies::class,
                IlluminateCookieMiddlewareAddQueuedCookiesToResponse::class,
                IlluminateSessionMiddlewareStartSession::class,
                // IlluminateSessionMiddlewareAuthenticateSession::class,
                IlluminateViewMiddlewareShareErrorsFromSession::class,
                AppHttpMiddlewareVerifyCsrfToken::class,
                IlluminateRoutingMiddlewareSubstituteBindings::class,
            ],
            
            'api' => [
                'throttle:60,1',
                'bindings',
            ],
        ];

这个核心类继承自他的父类IlluminateFoundationHttpKernel::class,核心类做了很多事情,它会将所有的中间件全部存储到一个指定的数组,方便内核调用及其他类调用。


namespace AppHttp;
    
use AppApiMiddlewareVerifyApiToken;
use IlluminateFoundationHttpKernel as HttpKernel;
    
class Kernel extends HttpKernel

回到起点

Laravel的启动经历了很繁琐的一个过程。这也是Laravel优雅的关键点。


$response = $kernel->handle(
    $request = IlluminateHttpRequest::capture()
);

$response->send();

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

将请求传入则完成了整个laravel的启动,至于结果的返回则有开发者自行通过控制器或其他可访问类返回。

致谢

感谢你看到这里,本篇文章源码解析靠个人理解。如有出入请拍砖。

希望本篇文章可以帮到你。谢谢

原文地址:https://segmentfault.com/a/1190000016522619

以上是关于Laravel源码解析之从入口开始的主要内容,如果未能解决你的问题,请参考以下文章

laravel5.5源码解析(入口应用的初始化)

Memcached源码分析之从SET命令开始说起

Laravel框架下路由的使用(源码解析)

Android笔记之从源码解析Handler中ThreadLocal的作用以及IntentService解析

Android笔记之从源码解析Handler中ThreadLocal的作用以及IntentService解析

Laravel源码解析--看看Lumen到底比Laravel轻在哪里