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 ...
随机推荐
- vue 开发笔记
vue 开发记录 marked 插件的使用 import marked from "marked"; import hljs from "highlight.js&quo ...
- 77%的Linux运维都不懂的内核问题
前言 之前在实习时,听了 OOM 的分享之后,就对 Linux 内核内存管理充满兴趣,但是这块知识非常庞大,没有一定积累,不敢写下,担心误人子弟,所以经过一个一段时间的积累,对内核内存有一定了解之后, ...
- 安装mongo可视化管理工具mongo admin
https://github.com/mrvautin/adminMongo github地址 安装要求下载下来,然后安装即可 中间出现了问题: 说是开了代理,可以关掉代理之后,然后把下载下来的删了, ...
- Theme Section
题目链接 #include <cstdio> #include <iostream> #include <cstring> using namespace std; ...
- Java中对象拷贝的两种方式
引用的拷贝 //引用拷贝 private static void copyReferenceObject(){ Person p = new Person(23, "zhang") ...
- 基于角色权限管理:rbac设计分析以及具体细节
权限管理---设计分析以及具体细节 说起权限我们大家都知道,不一样的角色会有不一样的权限. 比如就像学生管理系统一样,管理员,老师,学生之间的权限都是不一样的,那么展示的页面也是不一样的. 所以,我们 ...
- 在线获取键盘按键值(ascii码)工具
在线获取键盘按键值(ascii码)工具 http://www.bejson.com/othertools/keycodes/ 可以根据输入的值获取对应的键盘ascii码值
- NET高性能IO
System.IO.Pipelines: .NET高性能IO https://www.cnblogs.com/xxfy1/p/9290235.html System.IO.Pipelines是一个新的 ...
- EOJ Problem #3261 分词 trie + dp + 小剪枝
http://acm.ecnu.edu.cn/problem/3261/ 分词 Time limit per test: 1.0 seconds Time limit all tests: 1.0 s ...
- HDU 5883 F - The Best Path 欧拉通路 & 欧拉回路
给定一个图,要求选一个点作为起点,然后经过每条边一次,然后把访问过的点异或起来(访问一次就异或一次),然后求最大值. 首先为什么会有最大值这样的分类?就是因为你开始点选择不同,欧拉回路的结果不同,因为 ...