图解 Laravel 请求的完整生命周期

2020 年 7 月 30 日 CSDN

来源 | CSDN博客精选
头图 | 视觉中国

世间万物皆有生命周期,当我们使用任何工具时都需要理解它的工作原理,那么用起来就会得心应手,应用开发也是如此。解了它的原理,那么使用起来就会游刃有余。

Laravel是一套简洁的PHP Web开发框架(PHP Web Framework),今天,我们就来了解 一下Laravel 的生命周期。在此之前,我们先回顾一下PHP 的生命周期。


PHP 的生命周期


生命周期


当我们请求一个php文件时,PHP 为了完成这次请求,会发生5个阶段的生命周期切换:


  1. 模块初始化(MINIT),即调用 php.ini 中指明的扩展的初始化函数进行初始化工作,如 mysql 扩展。

  2. 请求初始化(RINIT),即初始化为执行本次脚本所需要的变量名称和变量值内容的符号表,如 $_SESSION变量。

  3. 执行该PHP脚本。

  4. 请求处理完成(Request Shutdown),按顺序调用各个模块的 RSHUTDOWN 方法,对每个变量调用 unset函数,如 unset $_SESSION 变量。

  5. 关闭模块(Module Shutdown) , PHP调用每个扩展的 MSHUTDOWN 方法,这是各个模块最后一次释放内存的机会。这意味着没有下一个请求了。


PHP 的运行模式


PHP两种运行模式是WEB模式、CLI模式。

  1. 当我们在终端敲入php这个命令的时候,使用的是CLI模式。

  2. 当使用Nginx或者别web服务器作为宿主处理一个到来的请求时,使用的是WEB模式。


WEB模式和CLI(命令行)模式很相似,区别是:

  1. WEB模式为了应对并发,可能采用多线程,因此生命周期1和5有可能只执行一次,下次请求到来时重复2-4的生命周期,这样就节省了系统模块初始化所带来的开销。

  2. CLI 模式会在每次脚本执行经历完整的5个周期,因为你脚本执行完不会有下一个请求。


可以看出PHP生命周期是很对称的。说了这么多,就是为了定位Laravel运行在哪里,没错,Laravel仅仅运行再 第三个阶段:

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


Laravel 的生命周期


Laravel 的生命周期从public\index.php开始,从public\index.php结束。


下面是 public\index.php的全部源码,更具体来说可以分为四步:
  
  
    
// 1require __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自动加载需要的类

  1. 文件载入composer生成的自动加载设置,包括所有你 composer require的依赖。

  2. 生成容器Container,Application实例,并向容器注册核心组件(HttpKernel,ConsoleKernel ,ExceptionHandler)(对应代码2,容器很重要,后面详细讲解)。

  3. 处理请求,生成并发送响应(对应代码3,毫不夸张的说,你99%的代码都运行在这个小小的handle 方法里面)。

  4. 请求结束,进行回调(对应代码4,还记得可终止中间件吗?没错,就是在这里回调的)。

Laravel 的请求步骤,我们不妨在详细一点:


第一步:注册加载composer自动生成的class loader

就是加载初始化第三方依赖。

第二步:生成容器 Container

并向容器注册核心组件,是从 bootstrap/app.php 脚本获取 Laravel 应用实例。

这一步是重点,处理请求,并生成发送响应。

请求被发送到 HTTP 内核或 Console 内核,这取决于进入应用的请求类型。

取决于是通过浏览器请求还是通过控制台请求。这里我们主要是通过浏览器请求。

HTTP 内核继承自 Illuminate\Foundation\Http\Kernel 类,该类定义了一个 bootstrappers 数组,这个数组中的类在请求被执行前运行,这些 bootstrappers 配置了错误处理、日志、检测应用环境以及其它在请求被处理前需要执行的任务。
  
  
    
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 内核还定义了一系列所有请求在处理前需要经过的 HTTP 中间件,这些中间件处理 HTTP 会话的读写、判断应用是否处于维护模式、验证 CSRF 令牌等等。
HTTP 内核的标志性方法 handle处理的逻辑相当简单:获取一个 Request,返回一个Response,把该内核想象作一个代表整个应用的大黑盒子,输入 HTTP 请求,返回 HTTP 响应。
第四步:将请求传递给路由

在Laravel基础的服务启动之后,就要把请求传递给路由了。路由器将会分发请求到路由或控制器,同时运行所有路由指定的中间件。

传递给路由是通过 Pipeline(管道)来传递的,但是Pipeline有一堵墙,在传递给路由之前所有请求都要经过,这堵墙定义在app\Http\Kernel.php中的$middleware数组中,没错就是中间件,默认只有一个CheckForMaintenanceMode中间件,用来检测你的网站是否暂时关闭。这是一个全局中间件,所有请求都要经过,你也可以添加自己的全局中间件。

然后遍历所有注册的路由,找到最先符合的第一个路由,经过它的路由中间件,进入到控制器或者闭包函数,执行你的具体逻辑代码。

所以,当请求到达你写的代码之前,Laravel已经做了大量工作,请求也经过了千难万险,那些不符合或者恶意的的请求已被Laravel隔离在外。


    
    
      
更多精彩推荐
     
     
       
中国开源激荡 20 年:IT 江湖,谁主沉浮?
☞每个创始人都需要了解的来自 Y Combinator 的 13 个见解
那个从深圳流水线去了纽约做程序员的女工,最近失业了
没想到!!Unicode 字符还能这样玩?
为什么说机器学习是预防欺诈的最佳工具?
区块链是工业4.0的领引者
点分享
点点赞
点在看
登录查看更多
0

相关内容

Laravel是一套简洁、优雅的PHP Web开发框架(PHP Web Framework)。它可以让你从面条一样杂乱的代码中解脱出来;它可以帮你构建一个完美的 Web App,而且每行代码都可以简洁、富于表达力。 Laravel 中文文档
【2020新书】高级Python编程,620页pdf
专知会员服务
236+阅读 · 2020年7月31日
【2020新书】实战R语言4,323页pdf
专知会员服务
101+阅读 · 2020年7月1日
【2020新书】使用高级C# 提升你的编程技能,412页pdf
专知会员服务
58+阅读 · 2020年6月26日
最新《自动微分手册》77页pdf
专知会员服务
102+阅读 · 2020年6月6日
算法与数据结构Python,369页pdf
专知会员服务
164+阅读 · 2020年3月4日
【干货】大数据入门指南:Hadoop、Hive、Spark、 Storm等
专知会员服务
96+阅读 · 2019年12月4日
tf.GradientTape 详解
TensorFlow
120+阅读 · 2020年2月21日
在K8S上运行Kafka合适吗?会遇到哪些陷阱?
DBAplus社群
9+阅读 · 2019年9月4日
PC微信逆向:两种姿势教你解密数据库文件
黑客技术与网络安全
16+阅读 · 2019年8月30日
PHP使用Redis实现订阅发布与批量发送短信
安全优佳
7+阅读 · 2019年5月5日
使用 Canal 实现数据异构
性能与架构
20+阅读 · 2019年3月4日
React Native 分包哪家强?看这文就够了!
程序人生
13+阅读 · 2019年1月16日
我是一个爬虫
码农翻身
12+阅读 · 2018年6月4日
Arxiv
4+阅读 · 2019年1月1日
Arxiv
3+阅读 · 2018年10月25日
Arxiv
9+阅读 · 2018年10月24日
Arxiv
4+阅读 · 2018年7月4日
VIP会员
相关VIP内容
【2020新书】高级Python编程,620页pdf
专知会员服务
236+阅读 · 2020年7月31日
【2020新书】实战R语言4,323页pdf
专知会员服务
101+阅读 · 2020年7月1日
【2020新书】使用高级C# 提升你的编程技能,412页pdf
专知会员服务
58+阅读 · 2020年6月26日
最新《自动微分手册》77页pdf
专知会员服务
102+阅读 · 2020年6月6日
算法与数据结构Python,369页pdf
专知会员服务
164+阅读 · 2020年3月4日
【干货】大数据入门指南:Hadoop、Hive、Spark、 Storm等
专知会员服务
96+阅读 · 2019年12月4日
相关资讯
tf.GradientTape 详解
TensorFlow
120+阅读 · 2020年2月21日
在K8S上运行Kafka合适吗?会遇到哪些陷阱?
DBAplus社群
9+阅读 · 2019年9月4日
PC微信逆向:两种姿势教你解密数据库文件
黑客技术与网络安全
16+阅读 · 2019年8月30日
PHP使用Redis实现订阅发布与批量发送短信
安全优佳
7+阅读 · 2019年5月5日
使用 Canal 实现数据异构
性能与架构
20+阅读 · 2019年3月4日
React Native 分包哪家强?看这文就够了!
程序人生
13+阅读 · 2019年1月16日
我是一个爬虫
码农翻身
12+阅读 · 2018年6月4日
Top
微信扫码咨询专知VIP会员