设为首页 收藏本站
查看: 738|回复: 0

[经验分享] Yii PHP 框架分析 (一)

[复制链接]

尚未签到

发表于 2015-8-28 11:38:10 | 显示全部楼层 |阅读模式
  Yii PHP 框架分析 (一)
作者:wdy
  http://hi.baidu.com/delphiss/blog/item/f7da86d787adb72506088b4b.html
  基于yii1.0.8的代码分析的。用了一个下午整理的,流水账,感兴趣的凑合着先看,国庆期间推出个整理修改版,然后再完成后两个部分(MVC和Yii的整体结构分析)。
  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;
   }
}
  $componen是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));
}

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-105487-1-1.html 上篇帖子: 详解PHP使用DOMDocument类遍历、增加、修改、删除XML节点操作 下篇帖子: php 利用phpexcel 导出数据
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表