1. 启动 网站的唯一入口程序 index.php : $yii=dirname(__FILE__)./../framework/yii.php;$config=dirname(__FILE__)./protected/config/main.php;// remove the following line when in production modedefined(YII_DEBUG) or define(YII_DEBUG,true);requi

1. 启动

网站的唯一入口程序 index.php :

$yii=dirname(__FILE__).’/../framework/yii.php’;

$config=dirname(__FILE__).’/protected/config/main.php’;

// remove the following line when in production mode

defined(‘YII_DEBUG’) or define(‘YII_DEBUG’,true);

require_once($yii);

Yii::createWebApplication($config)->run();

上面的require_once($yii) 引用出了后面要用到的全局类Yii,Yii类是YiiBase类的完全继承:

class Yii extends YiiBase

{

}

系统的全局访问都是通过Yii类(即YiiBase类)来实现的,Yii类的成员和方法都是static类型。

2. 类加载

Yii利用PHP5提供的spl库来完成类的自动加载。在YiiBase.php 文件结尾处

spl_autoload_register(array(‘YiiBase’,'autoload’));

将YiiBase类的静态方法autoload 注册为类加载器。 PHP autoload 的简单原理就是执行 new 创建对象或通过类名访问静态成员时,系统将类名传递给被注册的类加载器函数,类加载器函数根据类名自行找到对应的类文件并include 。

下面是YiiBase类的autoload方法:

public static function autoload($className)

{

// use include so that the error PHP file may appear

if(isset(self::$_coreClasses[$className]))

include(YII_PATH.self::$_coreClasses[$className]);

else if(isset(self::$_classes[$className]))

include(self::$_classes[$className]);

else

include($className.’.php’);

}

可以看到YiiBase的静态成员$_coreClasses 数组里预先存放着Yii系统自身用到的类对应的文件路径:

private static $_coreClasses=array(

‘CApplication’ => ‘/base/CApplication.php’,

‘CBehavior’ => ‘/base/CBehavior.php’,

‘CComponent’ => ‘/base/CComponent.php’,

)

非 coreClasse 的类注册在YiiBase的$_classes 数组中:

private static $_classes=array();

其他的类需要用Yii::import()将类路径导入PHP include paths 中,直接

include($className.’.php’)

3. CWebApplication的创建

回到前面的程序入口的 Yii::createWebApplication($config)->run();

public static function createWebApplication($config=null)

{

return new CWebApplication($config);

}

现在autoload机制开始工作了。

当系统 执行 new CWebApplication() 的时候,会自动

include(YII_PATH.’/base/CApplication.php’)

将main.php里的配置信息数组$config传递给CWebApplication创建出对象,并执行对象的run() 方法启动框架。

CWebApplication类的继承关系

CWebApplication -> CApplication -> CModule -> CComponent

$config先被传递给CApplication的构造函数

public function __construct($config=null)

{

Yii::setApplication($this);

// set basePath at early as possible to avoid trouble

if(is_string($config))

$config=require($config);

if(isset($config['basePath']))

{

$this->setBasePath($config['basePath']);

unset($config['basePath']);

}

else

$this->setBasePath(‘protected’);

Yii::setPathOfAlias(‘application’,$this->getBasePath());

Yii::setPathOfAlias(‘webroot’,dirname($_SERVER['SCRIPT_FILENAME']));

$this->preinit();

$this->initSystemHandlers();

$this->registerCoreComponents();

$this->configure($config);

$this->attachBehaviors($this->behaviors);

$this->preloadComponents();

$this->init();

}

Yii::setApplication($this); 将自身的实例对象赋给Yii的静态成员$_app,以后可以通过 Yii::app() 来取得。

后面一段是设置CApplication 对象的_basePath ,指向 proteced 目录。

Yii::setPathOfAlias(‘application’,$this->getBasePath());

Yii::setPathOfAlias(‘webroot’,dirname($_SERVER['SCRIPT_FILENAME']));

设置了两个系统路径别名 application 和 webroot,后面再import的时候可以用别名来代替实际的完整路径。别名配置存放在YiiBase的 $_aliases 数组中。

$this->preinit();

预初始化。preinit()是在 CModule 类里定义的,没有任何动作。

$this->initSystemHandlers() 方法内容:

/**

* Initializes the class autoloader and error handlers.

*/

protected function initSystemHandlers()

{

if(YII_ENABLE_EXCEPTION_HANDLER)

set_exception_handler(array($this,’handleException’));

if(YII_ENABLE_ERROR_HANDLER)

set_error_handler(array($this,’handleError’),error_reporting());

}

设置系统exception_handler和 error_handler,指向对象自身提供的两个方法。

4. 注册核心组件

$this->registerCoreComponents();

代码如下:

protected function registerCoreComponents()

{

parent::registerCoreComponents();

$components=array(

‘urlManager’=>array(

‘class’=>’CUrlManager’,

),

‘request’=>array(

‘class’=>’CHttpRequest’,

),

‘session’=>array(

‘class’=>’CHttpSession’,

),

‘assetManager’=>array(

‘class’=>’CAssetManager’,

),

‘user’=>array(

‘class’=>’CWebUser’,

),

‘themeManager’=>array(

‘class’=>’CThemeManager’,

),

‘authManager’=>array(

‘class’=>’CPhpAuthManager’,

),

‘clientScript’=>array(

‘class’=>’CClientScript’,

),

);

$this->setComponents($components);

}

注册了几个系统组件(Components)。

Components 是在 CModule 里定义和管理的,主要包括两个数组

private $_components=array();

private $_componentConfig=array();

每个 Component 都是 IApplicationComponent接口的实例,Componemt的实例存放在$_components 数组里,相关的配置信息存放在$_componentConfig数组里。配置信息包括Component 的类名和属性设置。

CWebApplication 对象注册了以下几个Component:urlManager,request,session,assetManager,user,themeManager,authManager,clientScript。

CWebApplication的parent 注册了以下几个Component:coreMessages,db,messages,errorHandler,securityManager,statePersister。

Component 在YiiPHP里是个非常重要的东西,它的特征是可以通过 CModule 的 __get() 和 __set() 方法来访问。 Component 注册的时候并不会创建对象实例,而是在程序里被第一次访问到的时候,由CModule 来负责(实际上就是 Yii::app())创建。

5. 处理 $config 配置

继续, $this->configure($config);

configure() 还是在CModule 里:

public function configure($config)

{

if(is_array($config))

{

foreach($config as $key=>$value)

$this->$key=$value;

}

}

实际上是把$config数组里的每一项传给 CModule 的 父类 CComponent __set() 方法。

public function __set($name,$value)

{

$setter=’set’.$name;

if(method_exists($this,$setter))

$this->$setter($value);

else if(strncasecmp($name,’on’,2)===0 && method_exists($this,$name))

{

//duplicating getEventHandlers() here for performance

$name=strtolower($name);

if(!isset($this->_e[$name]))

$this->_e[$name]=new CList;

$this->_e[$name]->add($value);

}

else if(method_exists($this,’get’.$name))

throw new CException(Yii::t(‘yii’,'Property “{class}.{property}” is read only.’,

array(‘{class}’=>get_class($this), ‘{property}’=>$name)));

else

throw new CException(Yii::t(‘yii’,'Property “{class}.{property}” is not defined.’,

array(‘{class}’=>get_class($this), ‘{property}’=>$name)));

}

我们来看看:

if(method_exists($this,$setter))

根据这个条件,$config 数组里的basePath, params, modules, import, components 都被传递给相应的 setBasePath(), setParams() 等方法里进行处理。

6、$config 之 import

其中 import 被传递给 CModule 的 setImport:

public function setImport($aliases)

{

foreach($aliases as $alias)

Yii::import($alias);

}

Yii::import($alias)里的处理:

public static function import($alias,$forceInclude=false)

{

// 先判断$alias是否存在于YiiBase::$_imports[] 中,已存在的直接return, 避免重复import。

if(isset(self::$_imports[$alias])) // previously imported

return self::$_imports[$alias];

// $alias类已定义,记入$_imports[],直接返回

if(class_exists($alias,false))

return self::$_imports[$alias]=$alias;

// 类似 urlManager 这样的已定义于$_coreClasses[]的类,或不含.的直接类名,记入$_imports[],直接返回

if(isset(self::$_coreClasses[$alias]) || ($pos=strrpos($alias,’.'))===false) // a simple class name

{

self::$_imports[$alias]=$alias;

if($forceInclude)

{

if(isset(self::$_coreClasses[$alias])) // a core class

require(YII_PATH.self::$_coreClasses[$alias]);

else

require($alias.’.php’);

}

return $alias;

}

// 产生一个变量 $className,为$alias最后一个.后面的部分

// 这样的:’x.y.ClassNamer’

// $className不等于 ‘*’, 并且ClassNamer类已定义的, ClassNamer’ 记入 $_imports[],直接返回

if(($className=(string)substr($alias,$pos+1))!==’*’ && class_exists($className,false))

return self::$_imports[$alias]=$className;

// 取得 $alias 里真实的路径部分并且路径有效

if(($path=self::getPathOfAlias($alias))!==false)

{

// $className!==’*',$className 记入 $_imports[]

if($className!==’*')

{

self::$_imports[$alias]=$className;

if($forceInclude)

require($path.’.php’);

else

self::$_classes[$className]=$path.’.php’;

return $className;

}

// $alias是’system.web.*’这样的已*结尾的路径,将路径加到include_path中

else // a directory

{

set_include_path(get_include_path().PATH_SEPARATOR.$path);

return self::$_imports[$alias]=$path;

}

}

else

throw new CException(Yii::t(‘yii’,'Alias “{alias}” is invalid. Make sure it points to an existing directory or file.’,array(‘{alias}’=>$alias)));

}

7. $config 之 components

$config 数组里的 $components 被传递给CModule 的setComponents($components)

public function setComponents($components)

{

foreach($components as $id=>$component)

{

if($component instanceof IApplicationComponent)

$this->setComponent($id,$component);

else if(isset($this->_componentConfig[$id]))

$this->_componentConfig[$id]=CMap::mergeArray($this->_componentConfig[$id],$component);

else

$this->_componentConfig[$id]=$component;

}

}

$component是IApplicationComponen的实例的时候,直接赋值:

$this->setComponent($id,$component),

public function setComponent($id,$component)

{

$this->_components[$id]=$component;

if(!$component->getIsInitialized())

$component->init();

}

如果$id已存在于_componentConfig[]中(前面注册的coreComponent),将$component 属性加进入。

其他的component将component属性存入_componentConfig[]中。

8. $config 之 params

这个很简单

public function setParams($value)

{

$params=$this->getParams();

foreach($value as $k=>$v)

$params->add($k,$v);

}

configure 完毕!

9. attachBehaviors

$this->attachBehaviors($this->behaviors);

空的,没动作

预创建组件对象

$this->preloadComponents();

protected function preloadComponents()

{

foreach($this->preload as $id)

$this->getComponent($id);

}

getComponent() 判断_components[] 数组里是否有 $id的实例,如果没有,就根据_componentConfig[$id]里的配置来创建组件对象,调用组件的init()方法,然后存入_components[$id]中。

10. init()

$this->init();

函数内:$this->getRequest();

创建了Reques 组件并初始化。

11. run()

public function run()

{

$this->onBeginRequest(new CEvent($this));

$this->processRequest();

$this->onEndRequest(new CEvent($this));

}

本文原创发布php中文网,转载请注明出处,感谢您的尊重!

查看 php yii脚本位置,Yii框架分析(一)入口脚本index.php的启动过程剖析相关推荐

  1. linux启动过程剖析,分析Linux系统的启动过程

    导读 一直使用linux系统,却对系统启动过程及系统初始化和各种服务的启动不太清楚.今天终于搞明白整个是怎么一回事了.本来想自己写篇文章,刚好在网上看到一篇不错的介绍,很详细,就直接拿来了. Linu ...

  2. yii+php+当前目录,Yii应用的目录结构和入口脚本

    以下是一个通过高级模版安装后典型的Yii应用的目录结构: ~~~ . ├── backend ├── common ├── console ├── environments ├── frontend ...

  3. Tampermonkey脚本编写笔记一:页面元素及框架分析

    我们在编写Tampermonkey脚本时,首先分析页面,定位元素,需要分以下三步: 1.定位元素:通过开发人员工具使用"审查元素",即可定位到页面中的元素 2.分析元素所在的窗口: ...

  4. Yii学习--使用Yii来建立博客

    Yii 之初体验 安装Yii 创建应用骨架 应用的工作流程 在这一部分里,我们将讲解怎样建立一个程序的骨架作为着手点.为简单起见,我们假设Web服务器根目录是/wwwroot ,相应的URL是 htt ...

  5. OpenGL研究, GUI框架分析, 虚拟机比较, Win10历险记, WxWidget, uboot, WireShark

    http://antkillerfarm.github.io/ OpenGL研究 书籍 我手上其实有几本关于OpenGL的实体书,但是比较了一下之后,发现还是电子版的<OpenGL编程指南> ...

  6. yii+php+当前目录,Yii常用路径方法总结

    Yii框架中的一些常见的路径方法的使用方法总结. 在控制器添加CSS文件或JavaScript文件: Yii::app()->clientScript->registerCssFile(Y ...

  7. (七) UVC框架分析

    title: UVC框架分析 date: 2019/4/23 19:50:00 toc: true --- UVC框架分析 源码的位置在drivers\media\video\uvc,查看下Makef ...

  8. Android DRM框架分析

    Android DRM框架分析 1. DRM框架 2.DRM架构 3.DRM插件 4. 实现 5.DRM插件详情 6.MediaDrm 7.MediaCrypto 8.参考链接 1. DRM框架 An ...

  9. Instrumentation框架分析及其使用

    本文旨在从Android系统源码出发,简单梳理Instrumentation框架的行为及逻辑结构,供有兴趣的同学一起学习 从am instrument谈起 am instrument命令的执行 我们知 ...

最新文章

  1. linux下occi操作oracle数据库,中文乱码的问题
  2. 人工智能 有信息搜索 (启发式)
  3. Windows SQL Server 2008 群集(摘自网络)
  4. 11种常见SQLMAP使用方法详解
  5. 项目中AppDelegate详解
  6. 云计算平台中虚拟专用网和VPC有什么区别?
  7. javaio流层次结构_流的多层次分组
  8. http 与https 区别浅析
  9. 计算机科学与技术实训内容,计算机科学与技术专业实训大纲.doc
  10. 如何寻找竞争情报发挥企业优势
  11. 12款好用超赞的国外搜索资源网站 ,开发者们的标配,你都知道吗?不知道就OUT了...
  12. 翼次元空间:智协云店通+BitCOO的4WiN.io全球互贸链 | Fund++
  13. RMS数据采集分布式架构
  14. Java流(Stream)操作实例-筛选、映射、查找匹配
  15. ORB-SLAM Spanning Tree 的作用
  16. 刘彬20000词汇03
  17. 2022年南京大学计算机拔尖班初试考后感想
  18. 【声卡宿主】ProTools 2021.7.0使用安装教程
  19. 3GPP TS 29244-g30 中英文对照 | 5.2.2 Usage Reporting Rule Handling
  20. Flex for .NET platform

热门文章

  1. 更改hadoop集群yarn的webui中的开始时间和结束时间为本地时间
  2. 垂直搜索引擎完整实现
  3. container_of宏定义分析---linux内核
  4. 汉语编程-现存的可能误区及可能方向思考
  5. manjaro软件源报错 不停看到错误 “PackageName: signature from “User <email@archlinux.org>“ is invalid“ 的几种解决方法
  6. spring的@primary和@qualifier注解解决一个接口多个实现的注入问题
  7. 大神干货:腾讯广告算法大赛亚军女极客生存图鉴
  8. Airbnb搜索:深度学习排序算法如何进化?
  9. 模型增强 | 利用 NLG 增强 QA 任务性能
  10. 机器学习中常用的优化方法