Laravel启动执行关键流程

一、 laravel结构

    |– app 包含Controller、Model、路由等在内的应用目录,大部分业务将在该目录下进行
    |  |– Console 命令行程序目录
    |  |  |– Commands 包含了用于命令行执行的类,可在该目录下自定义类
    |  |  |– Kernel.php 命令调用内核文件,包含commands变量(命令清单,自定义的命令需加入到这里)和schedule方法(用于任务调度,即定时任务)
    |  |– Events 事件目录
    |  |– Exceptions 包含了自定义错误和异常处理类
    |  |– Http HTTP传输层相关的类目录
    |  |  |– Controllers 控制器目录
    |  |  |– Middleware 中间件目录
    |  |  |– Requests 请求类目录
    |  |  |– Kernel.php 包含http中间件和路由中间件的内核文件
    |  |  |– routes.php 强大的路由
    |  |– Jobs 该目录下包含队列的任务类
    |  |– Listeners 监听器目录
    |  |– Providers 服务提供者目录
    |  |– User.php 自带的模型实例,我们新建的Model默认也存储在该目录
    |– bootstrap 框架启动载入目录
    |  |– app.php 创建框架应用实例
    |  |– autoload.php 自动加载
    |  |– cache 存放框架启动缓存,web服务器需要有该目录的写入权限
    |– config 各种配置文件的目录
    |  |– app.php 系统级配置文件
    |  |– auth.php 用户身份认证配置文件,指定好table和model就可以很方便地用身份认证功能了
    |  |– broadcasting.php 事件广播配置文件
    |  |– cache.php 缓存配置文件
    |  |– compile.php 编译额外文件和类需要的配置文件,一般用户很少用到
    |  |– database.php 数据库配置文件
    |  |– filesystems.php 文件系统配置文件,这里可以配置云存储参数
    |  |– mail.php 电子邮件配置文件
    |  |– queue.php 消息队列配置文件
    |  |– services.php 可存放第三方服务的配置信息
    |  |– session.php 配置session的存储方式、生命周期等信息
    |  |– view.php 模板文件配置文件,包含模板目录和编译目录等
    |– database 数据库相关目录
    |  |– factories 5.1以上版本的新特性,工厂类目录,也是用于数据填充
    |  |  |– ModelFactory.php 在该文件可定义不同Model所需填充的数据类型
    |  |– migrations 存储数据库迁移文件
    |  |– seeds 存放数据填充类的目录
    |     |– DatabaseSeeder.php 执行php artisan db:seed命令将会调用该类的run方法。该方法可调用执行该目录下其他Seeder类,也可调用factories方法生成ModelFactory里定义的数据模型
    |– public 网站入口,应当将ip或域名指向该目录而不是根目录。可供外部访问的css、js和图片等资源皆放置于此
    |  |– index.php 入口文件
    |  |– .htaccess Apache服务器用该文件重写URL
    |  |– web.config IIS服务器用该文件重写URL
    |– resources 资源文件目录
    |  |– assets 可存放包含LESS、SASS、CoffeeScript在内的原始资源文件
    |  |– lang 本地化文件目录
    |  |– views 视图文件就放在这啦
    |– storage 存储目录。web服务器需要有该目录及所有子目录的写入权限
    |  |– app 可用于存储应用程序所需的一些文件
    |  |– framework 该目录下包括缓存、sessions和编译后的视图文件
    |  |– logs 日志目录
    |– tests 测试目录
    |– vendor 该目录下包含Laravel源代码和第三方依赖包
    |– .env 环境配置文件。config目录下的配置文件会使用该文件里面的参数,不同生产环境使用不同的.env文件即可。
    |– artisan 强大的命令行接口,你可以在app/Console/Commands下编写自定义命令
    |– composer.json 存放依赖关系的文件
    |– composer.lock 锁文件,存放安装时依赖包的真实版本
    |– gulpfile.js gulp(一种前端构建工具)配置文件
    |– package.json gulp配置文件
    |– phpspec.yml phpspec(一种PHP测试框架)配置文件
    |– phpunit.xml phpunit(一种PHP测试框架)配置文件
    |– server.php PHP内置的Web服务器将把这个文件作为入口。以public/index.php为入口的可以忽略掉该文件

二、启动过程

index.php
bootstrap/autoload.php      --> 自动加载
bootstrap/app.php           --> 初始化服务容器(注册基础的服务提供者(事件、日志、路由)、注册核心类别名)
bootstrap/app.php           --> 注册共享的Kernel和异常处理器
Foundation\Http\Kernel.php  --> 处理请求和响应
index.php                   --> 将响应信息发送到浏览器
index.php                   --> 处理继承自TerminableMiddleware接口的中间件(Session)并结束应用生命周期

其中处理请求和响应包括:
    解析Illuminate\Contracts\Http\Kernel,实例化App\Http\Kernel
    a. 实例化Kernel : 构造函数:设置$app/$router,初始化$router中middleware数值
    b. handle处理请求:
        加载路由中间件、加载环境变量、加载配置文件、加载异常处理机制、注册门面、注册服务提供者、启动服务提供者、管道模式注入中间件
    c.将响应信息发送到浏览器
        注册request实例到容器 ($app['request']->Illuminate\Http\Request)  --  $request是经过Symfony封装的请求对象
        清空之前容器中的request实例
        调用bootstrap方法,启动一系列启动类的bootstrap方法:
        Illuminate\Foundation\Bootstrap\DetectEnvironment  环境配置($app[‘env’])
        Illuminate\Foundation\Bootstrap\LoadConfiguration  基本配置($app[‘config’])
        Illuminate\Foundation\Bootstrap\ConfigureLogging   日志文件($app[‘log’])
        Illuminate\Foundation\Bootstrap\HandleExceptions   错误&异常处理
        Illuminate\Foundation\Bootstrap\RegisterFacades    清除已解析的Facade并重新启动,注册config文件中alias定义的所有Facade类到容器
        Illuminate\Foundation\Bootstrap\RegisterProviders  注册config中providers定义的所有Providers类到容器
        Illuminate\Foundation\Bootstrap\BootProviders      调用所有已注册Providers的boot方法
        通过Pipeline发送请求,经过中间件,再由路由转发,最终返回响应

1.自动加载
    包括全局函数的加载、顶级命名空间映射、PSR0、PSR4标准的实现

2.初始化服务容器
    注册容器本身
        将基本的绑定注册到容器中,包括容器自身、容器实例名称app
        实例化
            app, Illuminate\Container\Container
        关键代码:
            protected function registerBaseBindings() {
                static::setInstance($this);
                $this->instance('app', $this);
                $this->instance(Container::class, $this);
            }
    注册基础服务提供者
        向容器分别注册了Key为以下值得实例
            events
            log
            router、url、redirect、Illuminate\Contracts\Routing\ResponseFactory
        关键代码:
            protected function registerBaseServiceProviders() {
                $this->register(new EventServiceProvider($this));
                $this->register(new LogServiceProvider($this));
                $this->register(new RoutingServiceProvider($this));
            }
    注册容器别名(注册共享的Kernel)
        在调用此方法之前,我们想取得一个容器实例的做法是 App::make('app');
        现在我们可以使用三种方法来取得一个容器实例app
                    App::make('Illuminate\Foundation\Application')
                    App::make('Illuminate\Contracts\Container\Container')
                    App::make('Illuminate\Contracts\Foundation\Application')
        关键代码:
            public function registerCoreContainerAliases(){
                ...
            }

3. 注册共享的Kernel和异常处理器
        关键代码:
            $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
            );

4. 处理请求和响应
        实例化App\Http\Kernel
        构造函数:设置$app/$router,初始化$router中middleware数值
        关键代码:
                public function __construct(Application $app, Router $router)
                {
                    $this->app = $app;
                    $this->router = $router;

                    $router->middlewarePriority = $this->middlewarePriority;

                    foreach ($this->middlewareGroups as $key => $middleware) {
                        $router->middlewareGroup($key, $middleware);
                    }

                    foreach ($this->routeMiddleware as $key => $middleware) {
                        $router->aliasMiddleware($key, $middleware);
                    }
                }

5. handle处理请求
        a. 注册request实例到容器 ($app['request']->Illuminate\Http\Request)  --  $request是经过Symfony封装的请求对象
        b. 清空之前容器中的request实例
        c. 调用bootstrap方法,启动一系列启动类的bootstrap方法
        d. 通过Pipeline发送请求,经过中间件,再由路由转发,最终返回响应

        关键代码:
            protected function sendRequestThroughRouter($request)
            {
                $this->app->instance('request', $request);

                Facade::clearResolvedInstance('request');

                $this->bootstrap();

                return (new Pipeline($this->app))
                            ->send($request)
                            ->through($this->app->shouldSkipMiddleware() ? [] : $this->middleware)
                            ->then($this->dispatchToRouter());
            }

6. bootstrap方法
        a.检测环境变量文件是否正常
        b.取得配置文件,即把/config/下的所有配置文件读取到容器(app()->make('config')可以查看所有配置信息)
        c.注册异常: set_error_handler,set_exception_handler, register_shutdown_function
        d.把/config/app.php里面的aliases项利用PHP库函数class_alias创建别名,从此,我们可以使用App::make('app')方式取得实例
        e.把/config/app.php里面的providers项,注册到容器
        f.运行容器中注册的所有的ServiceProvider中得boot方法

        关键代码:
            protected $bootstrappers = [
                \Illuminate\Foundation\Bootstrap\LoadEnvironmentVariables::class,
                \Illuminate\Foundation\Bootstrap\LoadConfiguration::class,
                \Illuminate\Foundation\Bootstrap\HandleExceptions::class,
                \Illuminate\Foundation\Bootstrap\RegisterFacades::class,
                \Illuminate\Foundation\Bootstrap\RegisterProviders::class,
                \Illuminate\Foundation\Bootstrap\BootProviders::class,
            ];

7. 将响应信息发送到浏览器
        关键代码:
            $response->send();

9. 处理继承自TerminableMiddleware
        关键代码:
            $kernel->terminate($request, $response);

10. Laravel路由
    $this->dispatchToRouter()
    --> $this->router->dispatch($request)
    --> $this->dispatchToRoute($request);  -- /Illuminate/Routing/Router.php
    --> $response = $this->runRouteWithinStack($route, $request);

    //干货来了
    protected function runRouteWithinStack(Route $route, Request $request)
    {
        // 取得routes.php里面的Middleware节点
        $middleware = $this->gatherRouteMiddlewares($route);
        //这个有点眼熟
        return (new Pipeline($this->container))
                ->send($request)
                ->through($middleware)    //执行上述的中间件
                ->then(function($request) use ($route)
                {
                    //不容易啊,终于到Controller类了
                    return $this->prepareResponse(
                        $request,
                        $route->run($request); //run控制器
                    );
                });
    }

原文地址:https://www.cnblogs.com/one-villager/p/9370810.html

时间: 2024-09-28 16:00:09

Laravel启动执行关键流程的相关文章

BOS项目 第11天(activiti工作流第三天,流程实例管理、项目中的用户和角色同步到activiti的用户和组表、设计物流配送流程、启动物流配送流程、组任务操作(查询、拾取)、个人任务操作(查询、办理))

BOS项目笔记 第11天 今天内容安排: 1.流程实例管理(查询.查看运行状态) 2.将bos系统中的用户和角色同步到activiti的用户和组表 3.设计物流配送流程 4.启动物流配送流程 5.组任务操作(查询.拾取) 6.个人任务操作(查询.办理) 1. 流程实例管理 1.1 查询流程实例列表 第一步:创建一个流程实例管理Action,提供list方法,查询流程实例列表数据 第二步:配置struts.xml 第三步:提供processinstance.jsp页面,展示列表数据 <s:iter

spark 启动job的流程分析

从WordCount开始分析 编写一个例子程序 编写一个从HDFS中读取并计算wordcount的例子程序: packageorg.apache.spark.examples importorg.apache.spark.SparkContext importorg.apache.spark.SparkContext._ objectWordCount{ defmain(args : Array[String]) { valsc = newSparkContext(args(0),"wordco

【nginx】【转】Nginx启动框架处理流程

Nginx启动过程流程图: ngx_cycle_t结构体: Nginx的启动初始化在src/core/nginx.c的main函数中完成,当然main函数是整个Nginx的入口,除了完成启动初始化任务以外,也必定是所有功能模块的入口之处.Nginx的初始化工作主要围绕一个类型为ngx_cycle_t类型的全局变量(cycle)展开. ngx_cycle_t结构体类型: typedef struct ngx_cycle_s ngx_cycle_t; struct ngx_cycle_s { voi

&lt;转&gt;深入理解Activity启动流程(一)–Activity启动的概要流程

本文原创作者:Cloud Chou. 原文地址:http://www.cloudchou.com/android/post-788.html Android中启动某个Activity,将先启动Activity所在的应用.应用启动时会启动一个以应用包名为进程名的进程,该进程有一个主线程,叫ActivityThread,也叫做UI线程. 本系列博客将详细阐述Activity的启动流程,这些博客基于Cm 10.1源码研究. 深入理解Activity启动流程(二)--Activity启动相关类的类图 深

浅析laravel路由执行原理

包头SEO:目前很多文章已经对Laravel的执行原理做了详细介绍,这里只是为了个人做一下简单记录 首先看入口 index.php 关键的执行函数就是 handle方法 ,但是前面的几个预处理函数,包括了整合框架的大知识点. 进入 require_once __DIR__.'/../bootstrap/app.php'; 发现 $app初始化了Application对象 $app = new Illuminate\Foundation\Application( $_ENV['APP_BASE_P

Android 输入管理服务启动过程的流程

接上一篇"Android窗口管理服务相关对象的创建流程",这两天分析了一下Android 输入管理服务启动过程的流程,流程图如下所示. 版权声明:本文为博主原创文章,未经博主允许不得转载.

薪资涨势成英国启动加息关键

最新数据显示,英国经济规模超过金融危机前水平已确定无疑,但与之对比鲜明的是薪资上涨缓慢,这也成为英国央行迟迟未启动加息,甚至不敢明确表态将很快加息的一个主要因素.此外,欧元区增长停滞和地缘政治风险因素也将令英国央行在启动加息时更加谨慎. 复苏势头巩固 英国国家统计局15日公布的修正数据显示,今年第二季度英国国内生产总值(GDP)同比增长3.2%,增幅比上个月公布的初值3.1%小幅上升,当季经济环比增长0.8%,和初值一致. 数据显示,二季度英国工业产值环比增长0.3%,服务业产值增长1.0%,建

pxe(预启动执行环境)

# pxe服务器 pxe:(preboot execute environment,预启动执行环境).是inter公司推出的一款通过网络来引导操作系统的协议.广泛应用于网吧无盘系统.只需要将无盘服务器维护好,就能保证网络中的无盘工作站(客户机能正常运行.)pxe本讲目标用centos系统部署一台pxe服务器,实现通过pxe网络来给pxe客户机手动或自动安装centos7操作系统.pxe服务器需要的支撑软件:DHCP.TFTP.syslinux(提供pxe引导程序文件pxelinux.0).文件共

laravel启动过程简单解析

:first-child{margin-top:0!important}img.plugin{box-shadow:0 1px 3px rgba(0,0,0,.1);border-radius:3px}iframe{border:0}figure{-webkit-margin-before:0;-webkit-margin-after:0;-webkit-margin-start:0;-webkit-margin-end:0}kbd{border:1px solid #aaa;-moz-bord