Yii2应用的运行过程
每一个框架都有一个入口脚本,Yii2也不例外。一般来说,对于Web应用的入口脚本是YiiBasePath/frontend/web目录下的index.php。
先观察这个文件:
<?php
defined('YII_DEBUG') or define('YII_DEBUG', true);
defined('YII_ENV') or define('YII_ENV', 'dev');
require__DIR__.'/../../vendor/autoload.php';
require__DIR__.'/../../vendor/yiisoft/yii2/Yii.php';
require__DIR__.'/../../common/config/bootstrap.php';
require__DIR__.'/../config/bootstrap.php';
$config=yii\helpers\ArrayHelper::merge(
require__DIR__.'/../../common/config/main.php',
require__DIR__.'/../../common/config/main-local.php',
require__DIR__.'/../config/main.php',
require__DIR__.'/../config/main-local.php'
);
(newyii\web\Application($config))->run();
从换行上看,可以分为4个部分,第一部分表示是否开启调试模式和开发模式,一般在开发模式要这样设置。第二部分引入第三方类加载器、Yii的类加载器,通用模组需要启动的组件、以及前端模组需要启动的组件,第三部分是合并通用、前端模组的主要、本地配置文件,一般后者会覆盖前者的配置,如果存在相同配置项的话。第四部分是启动应用,这里以匿名类的方式启动了yii\web\Application,并且传入了配置参数,然后调用它的run()方法来执行APP生命周期所定义的几个方法,APP的 开始->初始化->请求之前事件->处理请求->请求之后事件->发送响应->结束。好,这就是一个Yii应用执行的过程。
具体执行过程我们从run()方法入手来一点点剥洋葱。
<?php // yii\base\Application public function run()
{
try {
$this->state = self::STATE_BEFORE_REQUEST; // 1.应用开始,设置为开始状态
$this->trigger(self::EVENT_BEFORE_REQUEST); // 2.触发请求之前的事件,当请求发过来后就会触发
$this->state = self::STATE_HANDLING_REQUEST; // 3.设置为正在处理请求的状态
$response = $this->handleRequest($this->getRequest()); // 3.处理请求,得到一个响应(结果)
$this->state = self::STATE_AFTER_REQUEST; // 4.设置为请求之后的状态
$this->trigger(self::EVENT_AFTER_REQUEST); // 4.触发请求之后的事件
$this->state = self::STATE_SENDING_RESPONSE; // 5.设置为正在发送响应的状态
$response->send(); // 5.发送响应
$this->state = self::STATE_END; // 6.应用结束,设置为结束状态
return $response->exitStatus; // 返回应用推出状态码
} catch(ExitException $e) {
$this->end($e->statusCode, isset($response) ? $response : null);
return $e->statusCode;
}
}
通过解析发现少了一个步骤,关于应用的初始化,那是因为在构建Application对象的时候还执行了以下语句:
<?php
// yii\base\Application
publicfunction__construct($config = []) {
Yii::$app = $this; // 1.Yii的静态成员$app指向Application对象
static ::setInstance($this); // 2.设置当前请求的Application实例
$this->state = self::STATE_BEGIN; // 3.设置Application的状态为开始
$this->preInit($config); // 4.执行预备初始化
$this->registerErrorHandler($config); // 5.注册一个错误处理组件
Component::__construct($config); // 6.调用BaseObject的构造方法,Component继承自BaseObject,它会设置配置参数以及调用APP的init()方法
}
1、2两步都是在构建实例,第4步是APP的预备初始化,第6步是调用祖先类的初始化方法,它也包括init()方法的初始化。好,我们重点关注第4步,预备初始化。
<?php // yii\base\Application public
function preInit(&$config) //这是一个引用类型的形参,不用复制$config数据 {
if (!isset($config['id'])) { // 1.id配置项是必须的
throw new InvalidConfigException('The "id" configuration for the Applicationis required.');
} if (isset($config['basePath'])) { // 2.如果存在basePath参数,这个参数也是必须的
$this->setBasePath($config['basePath']); //则设置这个参数
unset($config['basePath']); //并删除$config中的那个参数
}
else {
throw new InvalidConfigException('The "basePath" configuration for the Applicationis required.');
} if (isset($config['vendorPath'])) { // 3.如果存在第三方组件目录
$this->setVendorPath($config['vendorPath']); //设置它
unset($config['vendorPath']); //从$config中删除它
}
else { // set"@vendor" $this->getVendorPath(); //否则设置一个默认vendor目录,一般在basePath下面
} if (isset($config['runtimePath'])) { // 4.如果存在一个runtime目录
$this->setRuntimePath($config['runtimePath']); //设置它到这个APP
unset($config['runtimePath']); //然后从$config中删除它
}
else { // set"@runtime" $this->getRuntimePath(); //否则设置一个默认runtime目录,一般在basePath目录下面
} if (isset($config['timeZone'])) { // 5.是否有时区字段
$this->setTimeZone($config['timeZone']); //如果有则设置它
unset($config['timeZone']); //然后从$config中删除它
}
elseif (!ini_get('date.timezone')) { //检查php.ini配置文件中是否设置了时区项
$this->setTimeZone('UTC'); //如果没有设置则初始化为国际标准时区
} if (isset($config['container'])) { // 6.检查是否有服务容器字段
$this->setContainer($config['container']); //如果有则设置它
unset($config['container']); //然后从$config中删除它
} // merge core components with custom components foreach($this->coreComponents() as $id => $component) { // 7.合并核心组件与自定义组件
if (!isset($config['components'][$id])) { //如果核心组件在自定义组件中不存在(去重)
$config['components'][$id] = $component; //把核心组件加入到$config中
}
elseif (is_array($config['components'][$id]) && !isset($config['components'][$id]['class'])) {
$config['components'][$id]['class'] = $component['class']; //把核心组件的类加入到自定义组件中
}
}
}
不难发现,它总共设置了7个属性,分别是应用ID,应用basePah,三方组件目录,runtime目录,时区,服务容器,自定义组件合并(组件)。
回到Application::__construct()方法的第6步,这里主要是调用了祖先类BaseObject的构造方法,它主要做了两个操作,一个是Yii::configure($this, $config),一个是执行init()方法,简单说就是后初始化。
关于后初始化的内容介绍在我的另一篇博文,《Yii2应用的初始化》。
初次阅读Yii2源代码,有任何问题欢迎讨论。
Yii2应用的运行过程的更多相关文章
- 江太公:javascript count(a)(b)(c)(d)运行过程思考
昨天,我弟抛给我一个js的题,使用类似标题那样的调用方法计算a*b*c*d以致无穷的实现方法.思考了半天,终于理清了它的运行过程,记录于下: 函数体: <!DOCTYPE html> &l ...
- JAVA - JAVA编译运行过程
Java编译原理 *.java→*.class→机器码 java编译器 (编译) → 虚拟机(解释执行) → 解释器(翻译) → 机器码 1.Java编译过程与c/c++编译过程不同 Java编译程 ...
- 孙鑫MFC学习笔记3:MFC程序运行过程
1.MFC中WinMain函数的位置在APPMODUL.cpp APPMODUL.cpp中是_tWinMain,其实_tWinMain是一个宏#define _tWinMain WinMain 2.全 ...
- HOWTO - Basic MSI安装包在安装运行过程中如何获取完整源路径
有朋友问到如何在一个Windows Installer安装包中获取安装包源路径,就是在安装包运行过程中动态获取*.msi所在完整路径. 这个问题分两类,如果我们的安装包只是一个*.msi安装文件,那么 ...
- DirectShow程序运行过程简析
这段时间一直在学习陆其明老师的<DirectShow开发指南>一书,书中对DirectShow的很多细节讲解清晰,但是却容易让人缺少对全局的把握.在学习过程中,整理了关于DirectSho ...
- Java Executor并发框架(二)剖析ThreadPoolExecutor运行过程
上一篇从整体上介绍了Executor接口,从上一篇我们知道了Executor框架的最顶层实现是ThreadPoolExecutor类,Executors工厂类中提供的newScheduledThrea ...
- 基础知识《零》---Java程序运行机制及运行过程
Java运行机制 Java虚拟机(Java Virtual Machine):Java虚拟机可以理解成一个以字节码为机器指令的CPU:对于不同的运行平台,有不同的虚拟机:Java虚拟机机制屏蔽了底层运 ...
- .net学习之.net和C#关系、运行过程、数据类型、类型转换、值类型和引用类型、数组以及方法参数等
1..net 和 C# 的关系.net 是一个平台,C#是种语言,C#语言可以通过.net平台来编写.部署.运行.net应用程序,C#通过.net平台开发.net应用程序2..net平台的重要组成FC ...
- ionic 运行过程中动态切换API服务器地址
ionic 运行过程中动态切换API服务器地址 keywords: ionic,phonegap,cordova,网络制式,动态切换,变更,API,服务器地址,$resource,localstora ...
随机推荐
- django导入导出excel实践
一.xlrd和xlwt模块介绍 xlrd模块提供在任何平台上从excel电子表格(.xls和.xlsx)中提取数据的功能,xlwt模块提供生成与Microsoft Excel 95 到2003版本兼容 ...
- EXP-00000: Message 0 not found; No message file for product=RDBMS, facility=EXP问题的解决方案
EXP-00000: Message 0 not found; No message file for product=RDBMS, facility=EXP 最近在服务器上准备做一个批处理,定时备份 ...
- php对数组操作的函数
array_reverse 以相反的顺序返回数组 array_unique 数组元素去重(只对一维数组有效) array_intersect两个或多个数组取交集 implode和explode也 ...
- Linux 下 FTP虚拟用户的使用配置
Linux下FTP虚拟用户的使用配置 Linux的FTP服务支持3种用户: 1.匿名帐户 2.本地帐户 3.虚拟用户 为什么要使用虚拟用户: 匿名帐户可以很好的保证FTP服务器的安全性,但是,对匿名用 ...
- easyui---datalist相关知识
datalist 笔记: class:class="easyui-datalist" //对应标准元素:ul 表格线:lines="true" 远程数据绑定: ...
- 熔断降级(Polly)
熔断降级(Polly) https://www.cnblogs.com/qhbm/p/9224307.html 一. 什么是熔断降级 熔断就是"保险丝".当出现某些状况时,切断服务 ...
- springmvc httprequest 使用@Autowired注解
springmvc httprequest 使用@Autowired注解我一直有个疑问,就是注解后每次的httprequest 是不是都一样的了,然后会不会引发多线程问题? 代码如下: import ...
- Mysql一个表编码的坑,mark一下
问题:一个sql执行很慢,5分钟左右,关键是最大的表是5万出头,另一张表不到5000原因:是两个表的字符集不同,导致匹配时,没有匹配到 解决办法:将两个表的字符集改成一样具体的命令: ALTER TA ...
- babel7中 corejs 和 corejs2 的区别
babel7中 corejs 和 corejs2 的区别 最近在给项目升级 webpack4 和 babel7,有一些改变但是变化不大.具体过程可以参考这篇文章 webpack4:连奏中的进化.只是文 ...
- 解决resteasy上传表单文件名乱码
Dubbo在2.6版本后合并了dubbox的resteasy代码后,可以支持rest风格的接口发布,但是在使用form表单上传文件的时候,获取的文件名称是乱码. 下面通过对源码分析一下原因,并提供一种 ...