yii2 beta版 执行流程

自动加载

1.composer的自动加载

	//composer的加载实现了四种方式,可以看看
require(__DIR__ . '/../../vendor/autoload.php');

composer的自动加载只加载其下载的组件....和yii框架的加载没有任何关系 2.yii的加载

	spl_autoload_register(['Yii', 'autoload'], true, true);
Yii::$classMap = include(__DIR__ . '/classes.php');
Yii::$container = new yii\di\Container;

这里使用spl_autoload_register把Yii::autoload注册成自动加载器,

  • 判断在不在$classMap中,如果有别名获取别名的文件路径
  • 如果不在$classMap中,则去除下划线,使用别名获取文件路径
  • 引入该文件,并判断是不是类或者接口或者trait
	public static function autoload($className)
{
if (isset(static::$classMap[$className])) {
$classFile = static::$classMap[$className];
if ($classFile[0] === '@') {
$classFile = static::getAlias($classFile);
}
} elseif (strpos($className, '\\') !== false) {
$classFile = static::getAlias('@' . str_replace('\\', '/', $className) . '.php', false);
if ($classFile === false || !is_file($classFile)) {
return;
}
} else {
return;
} include($classFile); if (YII_DEBUG && !class_exists($className, false) && !interface_exists($className, false) && !trait_exists($className, false)) {
throw new UnknownClassException("Unable to find '$className' in file: $classFile. Namespace missing?");
}
}

Yii::$classMap中保存的是命名空间和文件实际路径的对应表,默认加载了所有yii提供的类 Yii::$container 实现了依赖注入, Yii::$app 继承serviceLocal..只实现了单例组件,其中属性_components保存的是对象,_definitions保存的是类,匿名函数或者对象

引导前初始化配置

1.检查配置中模块ID是不是存在
2.检查配置中basePath,vendorPath,runtimePath,timeZone是否存在,并设置
3.向配置中添加核心组件配置:
base:log,view,formatter,i18n,mailer,urlManager,assetManager,security
web:request,response,session,user,errorHandler

4.如果启用了YII_ENABLE_ERROR_HANDLER错误处理,则会注册错误处理组件.然后从配置中卸载errorHandler,避免之后的引导再次添加组件.

	    ini_set('display_errors', false);  //注意这里...
set_exception_handler([$this, 'handleException']);
set_error_handler([$this, 'handleError']);
if ($this->memoryReserveSize > 0) {
$this->_memoryReserve = str_repeat('x', $this->memoryReserveSize);
}
register_shutdown_function([$this, 'handleFatalError']);

5.然后将配置中的值赋给applicaition

属性 说明 所在文件
$_events 使用on xxx 开通注册事件 yii\base\Component
$_behaviors 使用 as xxx开头注册行为 yii\base\Component
$_components 保存组件单例对象 yii\di\ServiceLocator
$_definitions 保存组件的原始数据 yii\di\ServiceLocator
$params 参数,是个数组 yii\base\Module
$id 模块ID yii\base\Module
$module 该模块的父模块 yii\base\Module
$layout 布局 yii\base\Module
$controllerMap 控制器地图 yii\base\Module
$controllerNamespace 控制器命名空间 yii\base\Module
$defaultRoute 默认路由 yii\base\Module
$_basePath 模块的根目录 yii\base\Module
$_viewPath 模块的视图目录 yii\base\Module
$_layoutPath 模块的布局目录 yii\base\Module
$_modules 子模块 yii\base\Module
$_instances 这个不常用,获取当前请求模块的实例 yii\base\Module
$controllerNamespace 'app\\controllers' yii\base\Application
$name 应用名称 yii\base\Application
$version 版本 yii\base\Application
$charset 字符编码.默认utf8 yii\base\Application
$language 语言,默认en-US yii\base\Application
$sourceLanguage 源语言 yii\base\Application
$controller 当前的控制器实例 yii\base\Application
$layout 布局 yii\base\Application
$requestedRoute 请求的路由 yii\base\Application
$requestedAction 请求的动作 yii\base\Application
$requestedParams 请求的参数 yii\base\Application
$extensions 扩展 yii\base\Application
$bootstrap 引导 yii\base\Application
$state 状态 yii\base\Application
$defaultRoute 默认路由 yii\web\Application
$catchAll 捕捉所有 yii\web\Application
$controller 当前控制器实例 yii\web\Application

引导阶段

  • 1.初始化request组件
  • 2.设置@webroot和@web别名
  • 3.加载扩展extensions,如果没有设置extensions会读取@vendor/yiisoft/extensions.php中的扩展并设置别名,如果扩展可引导,则执行其bootstrap方法-----本身来讲扩展概念没什么用处,因为还是依靠modules来加载这些扩展
  • 4.加载bootstrap中设置的组件和模块或者类,如果引导类实现了BootstrapInterface接口则执行bootstrap的方法

运行

  • 触发请求前事件EVENT_BEFORE_REQUEST,yii-debug中注册了该事件
  • 处理请求,如果catchAll不为空,则使用里面的路由.否则的话会解析获得路由和参数,
    然后执行动作action,返回响应Response,或者响应的data数据.

    		public function handleRequest($request)
    {
    if (empty($this->catchAll)) {
    list ($route, $params) = $request->resolve();
    } else {
    $route = $this->catchAll[0];
    $params = array_splice($this->catchAll, 1);
    }
    try {
    Yii::trace("Route requested: '$route'", __METHOD__);
    $this->requestedRoute = $route;
    $result = $this->runAction($route, $params);
    if ($result instanceof Response) {
    return $result;
    } else {
    $response = $this->getResponse();
    if ($result !== null) {
    $response->data = $result;
    } return $response;
    }
    } catch (InvalidRouteException $e) {
    throw new NotFoundHttpException($e->getMessage(), $e->getCode(), $e);
    }
    }
  • 动作的执行:首先会根据路由创建控制器.然后执行动作.
        public function runAction($route, $params = [])
    {
    $parts = $this->createController($route);
    if (is_array($parts)) {
    /* @var $controller Controller */
    list($controller, $actionID) = $parts;
    $oldController = Yii::$app->controller;
    Yii::$app->controller = $controller;
    $result = $controller->runAction($actionID, $params);
    Yii::$app->controller = $oldController; return $result;
    } else {
    $id = $this->getUniqueId();
    throw new InvalidRouteException('Unable to resolve the request "' . ($id === '' ? $route : $id . '/' . $route) . '".');
    }
    }

    创建控制器

    	public function createController($route)
    {
    //路由为空则使用默认路由
    if ($route === '') {
    $route = $this->defaultRoute;
    } // double slashes or leading/ending slashes may cause substr problem
    $route = trim($route, '/');
    if (strpos($route, '//') !== false) {
    return false;
    } //分解路由...前两个第一个可能是模块..第二个才是控制器
    if (strpos($route, '/') !== false) {
    list ($id, $route) = explode('/', $route, 2);
    } else {
    $id = $route;
    $route = '';
    } // module and controller map take precedence
    $module = $this->getModule($id);
    if ($module !== null) {
    //如果有模块,则继续往下面找.这里是递归...能一直查询子模块
    return $module->createController($route);
    } //查看控制器地图...一般控制器地图里的控制器..作为产品临时使用的页面
    if (isset($this->controllerMap[$id])) {
    $controller = Yii::createObject($this->controllerMap[$id], [$id, $this]); return [$controller, $route];
    } //执行到最后,没有子模块的时候...则将id加上剩余的路径(可能有多个/文件夹)
    if (($pos = strrpos($route, '/')) !== false) {
    $id .= '/' . substr($route, 0, $pos);
    $route = substr($route, $pos + 1);
    } $controller = $this->createControllerByID($id);
    if ($controller === null && $route !== '') {
    $controller = $this->createControllerByID($id . '/' . $route);
    $route = '';
    } return $controller === null ? false : [$controller, $route];
    }

    获取模块

    	public function getModule($id, $load = true)
    {
    //这一段是获取子模块的..在运行中是没有作用的..这一小部分可以当方法使用..
    if (($pos = strpos($id, '/')) !== false) {
    // sub-module
    $module = $this->getModule(substr($id, 0, $pos)); return $module === null ? null : $module->getModule(substr($id, $pos + 1), $load);
    } //这是返回模块
    if (isset($this->_modules[$id])) {
    if ($this->_modules[$id] instanceof Module) {
    return $this->_modules[$id];
    } elseif ($load) {
    Yii::trace("Loading module: $id", __METHOD__);
    if (is_array($this->_modules[$id]) && !isset($this->_modules[$id]['class'])) {
    $this->_modules[$id]['class'] = 'yii\base\Module';
    }
    /* @var $module Module */
    $module = Yii::createObject($this->_modules[$id], [$id, $this]);
    $module->setInstance($module);
    return $this->_modules[$id] = $module;
    }
    } return null;
    }
  • 触发请求后事件EVENT_AFTER_REQUEST,yii未注册事件
  • 响应发送

yii2 beta版 执行流程的更多相关文章

  1. {Django基础九之中间件} 一 前戏 二 中间件介绍 三 自定义中间件 四 中间件的执行流程 五 中间件版登陆认证

    Django基础九之中间件 本节目录 一 前戏 二 中间件介绍 三 自定义中间件 四 中间件的执行流程 五 中间件版登陆认证 六 xxx 七 xxx 八 xxx 一 前戏 我们在前面的课程中已经学会了 ...

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

    Yii2 源码分析  入口文件执行流程 1. 入口文件:web/index.php,第12行.(new yii\web\Application($config)->run()) 入口文件主要做4 ...

  3. [App Store Connect帮助]六、测试 Beta 版本(1)TestFlight Beta 版测试概述(iOS、Apple TVOS、watchOS)

    TestFlight Beta 版测试让您可以分发您 App 的 Beta 版构建版本给测试员并收集反馈.您可以在您的 App Store Connect 帐户中一次为至多 100 个 App 启用 ...

  4. Ubuntu 安装yii2 advanced版 遇到的坑

    1.安装 Composer https://www.yiichina.com/doc/guide/2.0/start-installation通过 Composer 安装 curl -sS https ...

  5. Mysql漂流系列(一):MySQL的执行流程

    MySQL的执行流程 MySQL的执行流程: MySQL的执行流程分析: 1.当我们请求mysql服务器的时候,MySQL前端会有一个监听,请求到了之后,服务器得到相关的SQL语句,执行之前(虚线部分 ...

  6. yii2 advanced版基础部分

    yii2 advanced版 一.目录结构 1.backend 和 frontend : 前后台入口,相当于是一个单独的Basic应用,有自己的 mvc 目录.配置文件目录.入口文件目录 2.cons ...

  7. Beta版测试报告

    Beta版测试报告 测试中发现的Bug: Version 2.0 Bug List 1. 在动态监测界面,若随便点击“开始”.“关闭”.“结束”.红叉,会出现不定式崩溃现象. 2. 处理空数据时可能会 ...

  8. Dalvik虚拟机java方法执行流程和Method结构体分析

    Method结构体是啥? 在Dalvik虚拟机内部,每个Java方法都有一个对应的Method结构体,虚拟机根据此结构体获取方法的所有信息. Method结构体是怎样定义的? 此结构体在不同的andr ...

  9. linux下无线鼠标驱动执行流程

    操作系统: debian 7.4(linux 3.2.54) 硬件: 一个无线鼠标.一个有线鼠标.usb集线器. 从淘宝上花了15块钱买了个无线鼠标,很好奇它的驱动程序是如何执行的. 首先将usb集线 ...

随机推荐

  1. Android 4.4 KitKat升级率已经接近18%(2014-07-09 07:29)

    腾讯数码讯(编 译:张秀梅)按照惯例, 每个月的第一个星期的星期一谷歌都会发布最新一期Android版本分布图.从去年十月末谷歌发布Android 4.4 KitKat以来,截止到目前为止Androi ...

  2. Linux内核的idle进程分析

    1. idle是什么 简单的说idle是一个进程,其pid号为 0.其前身是系统创建的第一个进程.也是唯一一个没有通过fork()产生的进程. 在smp系统中,每一个处理器单元有独立的一个执行队列,而 ...

  3. iOS:第三方数据库文件FMDB的使用

    第三方数据库FMDB •FMDB的使用:在sqlite的基础上,将sqlite中的函数进行封装产生的一个数据库文件. –FMDB的好处是对基本C库的封装,方便使用.同时还提供了多线程操作数据库带来的读 ...

  4. STL iterator和reverse_iterator

    先看一段代码: #include <iostream> #include <deque> #include <algorithm> #include <ite ...

  5. 如何解决Ubuntu与Windows双系统时间不同步

    导读 不知道有没朋友跟我一样是 Ubuntu 和 Windows 双系统?今天有朋友问到我,当他从 Ubuntu 系统重新启动到 Windows 时,会发现 Windows 中的时间变了,他问我有没办 ...

  6. 微软 Windows 10 将支持 8 英寸以下 ARM 平板设备

    2015 年 1 月 24 日,  9:32 下午 - 微软本周展示了 Windows 10 一系列新的改变,也包括首次公开展示的 Windows 10 手机版,但 ARM 平板并没有得到太多提及. ...

  7. C# int与string转化

    1.int-->string ; string s1 = a.ToString(); string s2 = Convert.ToString(a); 2.string -->int &q ...

  8. 异类的Javascript处理和解析URL的方式

    通常来说,我们使用Javascript处理和解析URL是使用location对象.在今天这个代码小技巧中,我们使用另外一个比较异类的方式处理和解析URL. 代码如下: function parseUR ...

  9. angularjs中ng-class的使用

    1.方法一 通过数据的双向绑定(不推荐) <!DOCTYPE html> <html ng-app="myApp"> <head> <me ...

  10. ibatis自定义数据类型在不支持中文的数据库存储汉字

    道理很简单,把gbk的汉字转换成iso编码存进数据库就可以了,读出来的时候把iso转换成gbk还原出原始的汉字. ibatis可以自定义类型处理器,在这里面做编码转换再适合不过了! sqlmap-co ...