Laravel工作原因:请求的完整生命周期
导读:当我们使用任何一件工具时,如果理解其内部原理,那么用起来就会非常得心应手,不再为一些小错误找不到所以然,工作就变得游刃有余。 本文为各位讲解的是Laravel的工作原因。通过对其解析,一切都不再那么神秘,我们会更加自信的构建其应用。
Laravel是一套简洁的PHP Web开发框架(PHP Web Framework),今天,我们就来了解 一下Laravel 的生命周期。在此之前,我们先回顾一下PHP 的生命周期。
PHP 的生命周期
生命周期
当我们请求一个php文件时,PHP 为了完成这次请求,会发生5个阶段的生命周期切换:
- 模块初始化(MINIT),即调用 php.ini 中指明的扩展的初始化函数进行初始化工作,如 mysql 扩展。
- 请求初始化(RINIT),即初始化为执行本次脚本所需要的变量名称和变量值内容的符号表,如 $_SESSION变量。
- 执行该PHP脚本。
- 请求处理完成(Request Shutdown),按顺序调用各个模块的 RSHUTDOWN 方法,对每个变量调用 unset函数,如 unset $_SESSION 变量。
- 关闭模块(Module Shutdown) , PHP调用每个扩展的 MSHUTDOWN 方法,这是各个模块最后一次释放内存的机会。这意味着没有下一个请求了。
PHP 的运行模式
- 当我们在终端敲入php这个命令的时候,使用的是CLI模式。
- 当使用Nginx或者别web服务器作为宿主处理一个到来的请求时,使用的是WEB模式。
- WEB模式为了应对并发,可能采用多线程,因此生命周期1和5有可能只执行一次,下次请求到来时重复2-4的生命周期,这样就节省了系统模块初始化所带来的开销。
- CLI 模式会在每次脚本执行经历完整的5个周期,因为你脚本执行完不会有下一个请求。

理解这些,你就可以优化你的 Laravel 代码,可以更加深入的了解 Laravel 的singleton(单例)。至少你知道了,每一次请求结束,PHP 的变量都会 unset,Laravel 的 singleton 只是在某一次请求过程中的singleton;你在 Laravel 中的静态变量也不能在多个请求之间共享,因为每一次请求结束都会 unset。理解这些概念,是写高质量代码的第一步,也是最关键的一步。因此记住,PHP是一种脚本语言,所有的变量只会在这一次请求中生效,下次请求之时已被重置,而不像Java静态变量拥有全局作用。
Laravel 的生命周期

// 1
require __DIR__.'/../bootstrap/autoload.php';
// 2
$app = require_once __DIR__.'/../bootstrap/app.php';
$kernel = $app->make(Illuminate\Contracts\Http\Kernel::class);
// 3
$response = $kernel->handle(
$request = Illuminate\Http\Request::capture()
);
$response->send();
// 4
$kernel->terminate($request, $response);
- 文件载入composer生成的自动加载设置,包括所有你 composer require的依赖。
- 生成容器Container,Application实例,并向容器注册核心组件(HttpKernel,ConsoleKernel ,ExceptionHandler)(对应代码2,容器很重要,后面详细讲解)。
- 处理请求,生成并发送响应(对应代码3,毫不夸张的说,你99%的代码都运行在这个小小的handle 方法里面)。
- 请求结束,进行回调(对应代码4,还记得可终止中间件吗?没错,就是在这里回调的)。

Laravel 的请求步骤,我们不妨在详细一点:
第一步:注册加载composer自动生成的class loader
第二步:生成容器 Container
这一步是重点,处理请求,并生成发送响应。
protected $bootstrappers = [
//注册系统环境配置 (.env)
'Illuminate\Foundation\Bootstrap\DetectEnvironment',
//注册系统配置(config)
'Illuminate\Foundation\Bootstrap\LoadConfiguration',
//注册日志配置
'Illuminate\Foundation\Bootstrap\ConfigureLogging',
//注册异常处理
'Illuminate\Foundation\Bootstrap\HandleExceptions',
//注册服务容器的门面,Facade 是个提供从容器访问对象的类。
'Illuminate\Foundation\Bootstrap\RegisterFacades',
//注册服务提供者
'Illuminate\Foundation\Bootstrap\RegisterProviders',
//注册服务提供者 `boot`
'Illuminate\Foundation\Bootstrap\BootProviders',
];
注意顺序:Facades 先于ServiceProviders,Facades也是重点,后面说,这里简单提一下,注册 Facades就是注册 config\app.php中的aliases
数组,你使用的很多类,如Auth,Cache,DB等等都是Facades;而ServiceProviders的register方法永远先于boot方法执行,以免产生boot方法依赖某个实例而该实例还未注册的现象。
HTTP 内核的标志性方法 handle处理的逻辑相当简单:获取一个 Request,返回一个Response,把该内核想象作一个代表整个应用的大黑盒子,输入 HTTP 请求,返回 HTTP 响应。
第四步:将请求传递给路由

作者:佚名
THE END