Yii2 源码分析  入口文件执行流程

1. 入口文件:web/index.php,第12行。(new yii\web\Application($config)->run())

  入口文件主要做4件事:

  1. 设置环境
  2. 加载自动加载
  3. 引入Web需要的配置
  4. 运行应用Application

 1 <?php
2
3 // comment out the following two lines when deployed to production
4 defined('YII_DEBUG') or define('YII_DEBUG', true);
5 defined('YII_ENV') or define('YII_ENV', 'dev');
6
7 require __DIR__ . '/../vendor/autoload.php';
8 require __DIR__ . '/../vendor/yiisoft/yii2/Yii.php';
9
10 $config = require __DIR__ . '/../config/web.php';
11
12 (new yii\web\Application($config))->run();

2. 接下来看运行应用Application做了哪些事情  

  1. 执行应用类的构造方法:__construct()

    a. 设置全局应用对象:Yii::$app
    b. 加载配置,检查配置是否正常配置并设置全局配置参数:如,检查应用id是否设置,检查basePath是否设置,检查缓存目录...,并设置对应的应用属性参数
    c. 注册全局错误异常处理机制
    d. 设置配置中对应对象的属性

 189     /**
190 * Constructor.
191 * @param array $config name-value pairs that will be used to initialize the object properties.
192 * Note that the configuration must contain both [[id]] and [[basePath]].
193 * @throws InvalidConfigException if either [[id]] or [[basePath]] configuration is missing.
194 */
195 public function __construct($config = [])
196 {
197 Yii::$app = $this;
198 static::setInstance($this);
199
200 $this->state = self::STATE_BEGIN;
201
202 $this->preInit($config);
203
204 $this->registerErrorHandler($config);
205
206 Component::__construct($config);
207 }

  2. 执行init()方法,执行引导:bootstrap

    a. 注册Yii扩展的组件,组件目录位于:根目录\vendor\yiisoft\extension/php
    b. 注册用户自定义配置文件main.php中属性boostrap配置的容器配置

267     /**
268 * {@inheritdoc}
269 */
270 public function init()
271 {
272 $this->state = self::STATE_INIT;
273 $this->bootstrap();
274 }

  3. 执行run()方法,执行应用请求流程,Yii处理请求逻辑应用的方法入口,应用的生命周期及对应的行为、事件处理函数等入口  

 374    /**
375 * Runs the application.
376 * This is the main entrance of an application.
377 * @return int the exit status (0 means normal, non-zero values mean abnormal)
378 */
379 public function run()
380 {
381 try {
382 $this->state = self::STATE_BEFORE_REQUEST;
383 $this->trigger(self::EVENT_BEFORE_REQUEST);
384
385 $this->state = self::STATE_HANDLING_REQUEST;
386 $response = $this->handleRequest($this->getRequest());
387
388 $this->state = self::STATE_AFTER_REQUEST;
389 $this->trigger(self::EVENT_AFTER_REQUEST);
390
391 $this->state = self::STATE_SENDING_RESPONSE;
392 $response->send();
393
394 $this->state = self::STATE_END;
395
396 return $response->exitStatus;
397 } catch (ExitException $e) {
398 $this->end($e->statusCode, isset($response) ? $response : null);
399 return $e->statusCode;
400 }
401 }

解析Yii执行run()的生命周期,在这个方法中可以看到有一些$this->state变量设置,这个变量主要用来记载Yii应用请求处于每个阶段的处理过程,例如上面的__construct(),init(),以及run()都相应的设置了Yii全局应用请求的状态

其中有:STATE_BEGIN(应用实例化阶段)、STATE_INIT(执行init方法初始化阶段)、STATE_BEFORE_REQUEST(执行请求前阶段)、STATE_HANDLING_REQUEST(处理请求中阶段)、STATE_AFTER_REQUEST(执行请求后阶段)、STATE_SENDING_RESPONSE(返回响应阶段)、STATE_END(应用结束输出阶段)

这里主要来分析run()方法中第385行:$response = $this->handleRequest($this->getRequest()); 处理请求中阶段:

1. 获取请求并传递给handRequest处理请求方法,handleRequest方法位于:\vendor\yiisoft\yii2\web\Application文件中
2. Application文件第80-99行,解析请求URL地址,获取请求路径,地址参数:82行 list($route, $params) = $request->resolve();
3. Application文件第100-116行,根据请求路径及参数,创建路径指向的路由控制器文件,执行方法并传入参数:103行 $result = $this->runAction($route, $params);
  a. \vendor\yiisoft\yii2\base\Module::runAction($route, $params),找到路由文件地址:522行 $parts = $this->createController($route)
  b. 分析路由控制器,及请求方法:list($controller, $actionID) = $parts
  c. 设置全局控制器
  d. 执行控制器中的方法体并传入参数:528行 $result = $controller->runAction($actionID, $params)
    I. \vendor\yiisoft\yii2\base\Controller::runAction($id, $params = []),第126-138行,方法体创建过程
      i. 创建方法体对象:创建方法体对象 第219行 createAction($id)
      ii. 正则匹配请求方法名是否符合规范
      iii. 方法名转化,以'-'切割,转换大小写
      iiii. 为方法名添加'action'前缀
      iiiii. 验证该方法名是否存在该类中
      iiiiii. 检查该方法是否属于公共方法
      iiiiiii. 执行方法体前  
    II. 执行模块前置行为(全局行为处理)
    III. 执行方法体前置行为(这里可验证方法权限、参数、跨域等行为)
    IIII. 执行方法体【业务逻辑,这里就是我们编写代码逻辑controller::action】
    IIIII. 执行方法体后置行为(这里可对返回数据进行验证处理)
    IIIIII. 执行模块后置行为(全局行为处理)
    IIIIIII. 获取以上执行方法体以及模块行为事件处理的最终结果并返回:第172行

这里分析下Yii怎么解析路由文件,执行路径:web\Application::handleRequest->runAction()103行 => base\Module::runAction->createController()522行 =>base\Module::createController()562行:

1. 解析地址,以'/'切割,根据route路由地址解析出id模块以及操作控制器路由地址route 第564-579行
2. 加载模块id对应的模块配置 第582-589行
3. 根据控制器路由地址,加载控制器文件 第591-602行,主要代码:$controller = $this->createControllerByID($id . '/', $route) 598行
  a. 检查类文件及类前缀路径是否符合规范 vendor\yiisoft\base\Module::629行
  b. 加载模块完,加载控制器,检查类文件是否存在,控制器地址文件不存在则报错404
  c. 加载容器的时候,自动给方法加上controller vendor\yiisoft\base\Module.php::635行
  d. 匹配正确的文件,实例化控制器 vendor\yiisoft\base\Module.php::642行
4. 返回对应的类实例化及操作方法到执行前创建控制器 vendor\yiisoft\yii2\base\Module::522行
5. 检查返回是否符合数组(控制器实例及操作方法),否:页面找不到
6. 设置App应用全局控制器 vendor\yiisoft\yii2\base\Module::527行
7. 传递参数到控制器并执行控制器方法 vendor\yiisoft\yii2\base\Module::528行

Yii2 源码分析 入口文件执行流程的更多相关文章

  1. asyncio源码分析之基本执行流程

    基于async关键字的原生协程 # 定义一个简单的原生协程cor async def cor(): print('enter cor') print('exit cor') print(type(co ...

  2. mybatis(五):源码分析 - mapper文件解析流程

  3. MyBatis 源码分析 - 映射文件解析过程

    1.简介 在上一篇文章中,我详细分析了 MyBatis 配置文件的解析过程.由于上一篇文章的篇幅比较大,加之映射文件解析过程也比较复杂的原因.所以我将映射文件解析过程的分析内容从上一篇文章中抽取出来, ...

  4. (转)linux内存源码分析 - 内存回收(整体流程)

    http://www.cnblogs.com/tolimit/p/5435068.html------------linux内存源码分析 - 内存回收(整体流程) 概述 当linux系统内存压力就大时 ...

  5. JVM源码分析之JVM启动流程

      原创申明:本文由公众号[猿灯塔]原创,转载请说明出处标注 “365篇原创计划”第十四篇. 今天呢!灯塔君跟大家讲: JVM源码分析之JVM启动流程 前言: 执行Java类的main方法,程序就能运 ...

  6. Solr4.8.0源码分析(5)之查询流程分析总述

    Solr4.8.0源码分析(5)之查询流程分析总述 前面已经写到,solr查询是通过http发送命令,solr servlet接受并进行处理.所以solr的查询流程从SolrDispatchsFilt ...

  7. HDFS源码分析DataXceiver之整体流程

    在<HDFS源码分析之DataXceiverServer>一文中,我们了解到在DataNode中,有一个后台工作的线程DataXceiverServer.它被用于接收来自客户端或其他数据节 ...

  8. Yii2源码分析(一):入口

    写在前面,写这些随笔是记录下自己看Yii2源码的过程,可能会有些流水账,大部分解析放在注释里说明,由于个人水平有限,有不正确的地方还望斧正. web入口文件Index.php // 定义全局的常量,Y ...

  9. MyBatis源码分析-SQL语句执行的完整流程

    MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以对配置和原生Map使用简 ...

随机推荐

  1. UltraSoft - Alpha - Scrum Meeting 5

    Date: Apr q9th, 2020. Scrum 情况汇报 进度情况 组员 负责 昨日进度 后两日任务 CookieLau PM 统筹个人进度,协助推进进度 辅助前后端连接工作 刘zh 前端 完 ...

  2. C++ 、Qt计算时间的方法

    原文链接:https://blog.csdn.net/chy555chy/article/details/53405072 Qt计算时间的两种方法: QTime elapsed() : ms QTim ...

  3. 计算机网络之应用层概述(C/S模型与p2p模型)

    文章转自:https://blog.csdn.net/weixin_43914604/article/details/105582318 学习课程:<2019王道考研计算机网络> 学习目的 ...

  4. Centos7 升级过内核 boot分区无法挂载修

    参考连接:https://www.cnblogs.com/heqiuyong/p/11186301.html 故障图 挂载系统盘,光盘启动,急救模式, chroot /mnt/sysimage 报错 ...

  5. OpenAPITools 实践

    OpenAPITools 可以依据 REST API 描述文件,自动生成服务端桩(Stub)代码.客户端 SDK 代码,及文档等.其是社区版的 Swagger ,差异可见:OpenAPI Genera ...

  6. C++11 多线程同步 互斥锁 条件变量

    在多线程程序中,线程同步(多个线程访问一个资源保证顺序)是一个非常重要的问题,Linux下常见的线程同步的方法有下面几种: 互斥锁 条件变量 信号量 这篇博客只介绍互斥量和条件变量的使用. 互斥锁和条 ...

  7. maven项目打包执行

    1.maven项目已完成状态! 2.maven打包前提pom.xml引入插件依赖 <build> <plugins> <plugin> <artifactId ...

  8. C++ 函数模板实现原理剖析

    C++ 函数模板实现机制原理剖析 重点 编译器并不是把函数模板处理成能够处理任意类的函数 编译器从函数模板通过具体类型来产生不同的函数 编译器会对函数模板进行两次编译 (1)在声明的位置对模板代码进行 ...

  9. python编写脚本,登录Github通过指定仓库指定敏感关键字搜索自动化截图生成文件【完美截图】

    前言:为了避免开发人员将敏感信息写入文件传到github,所以测试人员需要检查每个仓库是否有写入,人工搜索审核比较繁琐,所以写一个脚本通过配置 配置文件,指定需要搜索的仓库和每个仓库需要搜索的关键字, ...

  10. Jenkins执行 remote SSH 命令

    1.安装 SSH Pipeline Steps 插件 2.在凭据中添加remote server凭据,如下 3.Pipeline编写: def GetRemoteServer(ip){ def rem ...