跟我学Laravel之请求(Request)的生命周期
概述
在现实世界中使用工具时,如果理解了工具的工作原理,使用起来就会更加有底气。应用开发也是如此。当你理解了开发工具是如何工作的,使用起来就会更加自如。这篇文档的目标就是提供一个高层次的概述,使你对于Laravel框架的运行方式有一个较好的把握。在更好地了解了整个框架之后,框架的组件和功能就不再显得那么神秘,开发起应用来也更加得心应手。这篇文档包含了关于请求生命周期的高层次概述,以及启动文件和应用程序事件的相关内容。
如果你不能立即理解所有的术语,别灰心,可以先有一个大致的把握,在阅读文档其他章节的过程中继续积累和消化知识。
请求的生命周期
发送给应用程序的所有请求都经由public/index.php脚本处理。如果使用的是Apache服务器,Laravel中包含的.htaccess文件将对所有请求进行处理并传递给index.php。这是Laravel从接受客户端请求到返回响应给客户端的整个过程的开始。若能对于Laravel的引导过程(bootstrapprocess)有一个大致的认识,将有助于理解框架,我们不妨先讨论这个。
到目前为止,学习Laravel引导过程所需掌握的最重要的概念就是服务提供器。打开app/config/app.php配置文件,找到providers数组,你会发现一个服务提供器的列表。这些提供器充当了Laravel的主要引导机制。在我们深入服务提供器之前,先回到index.php的讨论。当一个请求进入index.php文件,bootstrap/start.php文件会被加载。这个文件会创建一个LaravelApplication对象,该对象同时作为框架的IoC容器。
Application对象创建完成后,框架会设置一些路径信息并运行环境检测。然后会执行位于Laravel源码内部的引导脚本,并根据你的配置文件设置时区、错误报告等其他信息。除了配置这些琐碎的配置选项以外,该脚本还会做一件非常重要的事情:注册所有为应用程序配置的服务提供器。
简单的服务提供器只包含一个方法:register。当应用程序对象通过自身的register方法注册某个服务提供器时,会调用该服务提供器的register方法。服务提供器通过这个方法向IoC容器注册一些东西。从本质上讲,每个服务提供器都是将一个或多个闭包绑定到容器中,你可以通过这些闭包访问绑定到应用程序的服务。例如,QueueServiceProvider注册了多个闭包以便使用与队列相关的多个类。当然,服务提供器并不局限于向IoC容器注册内容,而是可以用于任何引导性质的任务。服务提供器可以注册事件监听器、视图合成器、Artisan命令等等。
在注册完所有服务提供器后,app/start下的文件会被加载。最后,app/routes.php文件会被加载。一旦routes.php文件被加载,Request对象就被发送给应用程序对象,继而被派发到某个路由上。
我们总结一下:
请求进入public/index.php文件。
bootstrap/start.php文件创建应用程序对象并检测环境。
内部的framework/start.php文件配置相关设置并加载服务提供器。
加载应用程序app/start目录下的文件。
加载应用程序的app/routes.php文件。
将Request对象发送给应用程序对象,应用程序对象返回一个Response对象。
将Response对象发回客户端。
你应该已经掌握了Laravel应用程序是如何处理发来的请求的。下面我们来看一下启动文件。
启动文件
应用程序的启动文件被存放在app/start目录中。默认情况下,该目录下包含三个文件:global.php、local.php和artisan.php文件。需要获取更多关于artisan.php的信息,可以参考文档Artisan命令行。
global.php启动文件默认包含一些基本项目,例如日志的注册以及载入app/filters.php文件。然而,你可以在该文件里做任何你想做的事情。无论在什么环境下,它都将会被自动包含进_每一个_request中。而local.php文件仅在local环境下被执行。获取更多关于环境的信息,请查看文档配置。
当然,如果除了local环境你还有其他环境的话,你也可以为针对这些环境创建启动文件。这些文件将在应用程序运行在该环境中时被自动包含。假设你在bootstrap/start.php文件中配置了一个development环境,你可以创建一个app/start/development.php文件,在那个环境下任何进入应用程序的请求都会包含该文件。
启动文件里存放什么
启动文件主要用来存放任何“引导”性质的代码。例如,你可以在启动文件中注册视图合成器,配置日志信息,或是进行一些PHP设置等。具体做什么取决于你。当然了,把所有引导代码都丢到启动文件里会使启动文件变得杂乱。对于大型应用而言,或是启动文件显得太杂乱了,请考虑将某些引导代码移至服务提供器中。
应用程序事件
你还可以通过注册before、after、finish和shutdown应用程序事件以便在处理request之前或后做一些操作:
注册应用程序事件
App::before(function($request) { // });
App::after(function($request,$response) { // });