>php教程 >PHP开发 >Yii 프레임워크 분석(3) - 클래스 로딩 메커니즘 및 관리, 구성, 액세스 및 애플리케이션 구성 요소 생성

Yii 프레임워크 분석(3) - 클래스 로딩 메커니즘 및 관리, 구성, 액세스 및 애플리케이션 구성 요소 생성

黄舟
黄舟원래의
2016-12-27 11:11:261443검색

Yii 애플리케이션의 입력 스크립트는 Yii 클래스를 참조합니다. Yii 클래스의 정의:

class Yii extends YiiBase
{
}

yiic이 만든 애플리케이션에서 Yii 클래스는 YiiBase의 "조끼"일 뿐입니다. 필요에 따라 나만의 Yii 클래스를 맞춤 설정할 수도 있습니다.

Yii(예: YiiBase)는 전체 애플리케이션에 대한 정적 및 전역 액세스를 제공하는 "헬퍼 클래스"입니다.

Yii 클래스의 여러 정적 멤버:
$_aliases: 시스템 별칭에 해당하는 실제 경로를 저장합니다
$_imports:
$_classes:
$_includePaths php 포함 경로
$_app: Yii::app()을 통해 액세스되는 CWebApplication 개체
$_logger: 시스템 로그 개체
$_app 개체는 Yii::createWebApplication() 메서드에 의해 생성됩니다.

클래스 자동 로딩

Yii는 php5의 자동 로딩 메커니즘을 기반으로 클래스 자동 로딩 기능을 제공합니다. 오토로더는 YiiBase 클래스의 정적 메소드인 autoload()입니다.

프로그램이 new를 사용하여 객체를 생성하거나 클래스의 정적 멤버에 액세스할 때 PHP는 클래스 이름을 클래스 로더에 전달하고 클래스 로더는 클래스 파일 포함을 완료합니다.

자동 로드 메커니즘은 클래스의 "요청 시 가져오기"를 구현합니다. 즉, 시스템은 클래스에 액세스할 때만 클래스의 파일을 포함합니다.

YiiBase 클래스의 정적 멤버인 $_coreClasses에는 Yii의 자체 코어 클래스 이름과 해당 클래스 파일 경로가 미리 저장되어 있습니다. 다른 Yii 응용 프로그램에서 사용되는 클래스는 Yii::import()를 사용하여 가져올 수 있습니다. Yii::import()는 단일 클래스와 해당 클래스 파일을 $_classes에 저장하고 * 와일드카드 문자로 표시된 경로를 php include_path에 추가합니다. Yii::import()로 가져온 클래스 파일이나 디렉터리는 여러 번 가져오는 것을 방지하기 위해 $_imports에 기록됩니다.

/* Yii::import()
* $alias: 要导入的类名或路径
* $forceInclude false:只导入不include类文件,true则导入并include类文件
*/
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) || interface_exists($alias,false))
        return self::$_imports[$alias]=$alias;
    // 已定义于$_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) || interface_exists($className,false)))
        return self::$_imports[$alias]=$className;
    // $alias里含有别名,并转换真实路径成功
    if(($path=self::getPathOfAlias($alias))!==false)
    {
        // 不是以*结尾的路径(单类)
        if($className!==’*')
        {
            self::$_imports[$alias]=$className;
            if($forceInclude)
                require($path.’.php’);
            else
                // 类名与真实路径记入$_classes数组
                self::$_classes[$className]=$path.’.php’;
            return $className;
        }
        // $alias是’system.web.*’这样的已*结尾的路径,将路径加到include_path中
        else // a directory
        {
            if(self::$_includePaths===null)
            {
                self::$_includePaths=array_unique(explode(PATH_SEPARATOR,get_include_path()));
                if(($pos=array_search(‘.’,self::$_includePaths,true))!==false)
                    unset(self::$_includePaths[$pos]);
             }
            array_unshift(self::$_includePaths,$path);
            set_include_path(‘.’.PATH_SEPARATOR.implode(PATH_SEPARATOR,self::$_includePaths));
            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)));
}

그런 다음 YiiBase::autoload() 함수의 처리를 살펴보세요.

public static function autoload($className)
{
    // $_coreClasses中配置好的类直接引入
    if(isset(self::$_coreClasses[$className]))
        include(YII_PATH.self::$_coreClasses[$className]);
    // $_classes 中登记的单类直接引入
    else if(isset(self::$_classes[$className]))
        include(self::$_classes[$className]);
    else
    {
        // 其他的认为文件路径以记入 include_path 里,以$className.’.php’直接引入
        include($className.’.php’);
        return class_exists($className,false) || interface_exists($className,false);
    }
    return true;
}

시스템 구성 파일의 가져오기 항목에 있는 클래스 또는 경로는 다음과 같습니다. 스크립트 시작 가져오기 중에 자동으로 로드됩니다. 사용자 애플리케이션의 개별 클래스로 가져와야 하는 클래스의 경우 클래스 정의 앞에 Yii::import() 문을 추가할 수 있습니다.

애플리케이션 컴포넌트 관리

앞서 언급했듯이 Yii의 CComponent 클래스는 컴포넌트 속성, 이벤트 및 동작에 대한 액세스 인터페이스를 제공하며 CComponent의 하위 클래스인 CModule은 애플리케이션 컴포넌트 관리도 제공합니다.

애플리케이션 구성 요소는 IApplicationComponent 인터페이스의 인스턴스여야 하며 인터페이스의 init() 및 getIsInitialized() 메서드를 구현해야 합니다. init()는 구성요소 초기화 매개변수를 적용한 후 자동으로 호출됩니다.

Yii 자체 기능 모듈은 공통 Yii::app()->user, Yii::app()->request 등 애플리케이션 컴포넌트를 통해 제공됩니다. 사용자는 애플리케이션 구성요소를 정의할 수도 있습니다.

Yii::app() 객체(CWebApplication)의 상위 클래스인 CModule은 구성 요소 생성, 초기화, 객체 저장 등을 포함한 완전한 구성 요소 수명 주기 관리를 제공합니다.

각 애플리케이션 구성 요소는 문자열 이름으로 식별되며 CModule 클래스의 __get() 메서드를 통해 액세스됩니다.

CModule 클래스의 $_comComponents[] 멤버는 애플리케이션 컴포넌트의 객체 인스턴스($name => $object)를 저장하고, $_comComponentConfig[]는 애플리케이션 컴포넌트의 클래스 이름과 초기화 매개변수를 저장합니다.

애플리케이션 컴포넌트를 사용하는 경우 먼저 $_comComponentConfig에 해당 컴포넌트의 클래스 이름과 초기화 매개변수를 설정합니다. 해당 컴포넌트에 처음 액세스하면 CModule이 자동으로 애플리케이션 컴포넌트 객체 인스턴스를 생성하고 지정된 매개변수를 초기화합니다. 그런 다음 애플리케이션 구성요소의 init() 메소드가 호출됩니다.

Yii::app() 객체의 CWebApplication 클래스와 해당 상위 클래스 CApplication은 시스템 자체에서 사용하는 애플리케이션 구성 요소(urlManager, request, session, AssetManager, user, themeManager, authManager)로 사전 구성되어 있습니다. , clientScript, coreMessages, db, 메시지, errorHandler, securityManager, statePersister.

시스템 구성 파일의 구성 요소 프로젝트에서 시스템 애플리케이션 구성 요소의 매개변수를 수정하거나 새 애플리케이션 구성 요소를 구성할 수 있습니다.

CModule은 애플리케이션 구성 요소 인스턴스 생성을 담당하지 않지만 Yii::createComponent() 정적 메서드에 의해 수행됩니다.

createComponent()의 $config 매개변수는 클래스 이름의 문자열이거나 클래스 이름과 초기화 매개변수를 저장하는 배열일 수 있습니다.

애플리케이션 구성 요소 구성

애플리케이션 구성 요소의 구성은 시스템 $config 변수(config/main.php)의 구성 요소 항목에 저장됩니다.

// application components
‘components’=>array(
    ‘log’=>array(
        ‘class’=>’CLogRouter’,
        ‘routes’=>array(
            array(
                ‘class’=>’CFileLogRoute’,
                ‘levels’=>’error, warning’,
            ),
         ),
    ),
    ‘user’=>array(
    // enable cookie-based authentication
        ‘allowAutoLogin’=>true,
    ),
),

$ config의 구성 요소 항목은 CApplication 생성자에서 처리됩니다.

$this->configure($config);

configure() 함수의 처리는 매우 간단합니다.

public function configure($config)
{
    if(is_array($config))
    {
        foreach($config as $key=>$value)
            $this->$key=$value;
    }
}

$config의 각 항목은 $_app 객체의 setXXX() 속성 설정 메소드에 속성으로 전달되며, 여기서 '컴포넌트' 항목은 CModule의 상위 클래스인 setComponents()에 의해 처리됩니다. CWebApplication에서.

setComponents()는 '컴포넌트' 항목의 클래스 이름과 초기화 매개변수를 $_comComponentConfig[]에 저장합니다.

public function setComponents($components)
{
    // $config 里的’components’每一项
    foreach($components as $id=>$component)
    {
        if($component instanceof IApplicationComponent)
            $this->setComponent($id,$component);
        // $_componentConfig里已经存在配置,合并$component
        else if(isset($this->_componentConfig[$id]))
            $this->_componentConfig[$id]=CMap::mergeArray($this->_componentConfig[$id],$component);
        // 在$_componentConfig里新建项目
        else
            $this->_componentConfig[$id]=$component;
    }
}

애플리케이션 구성 요소에 대한 액세스

CModule类重载了CComponent的__get()方法,优先访问应用组件对象。

public function __get($name)
{
    if($this->hasComponent($name))
        return $this->getComponent($name);
    else
        return parent::__get($name);
}

hasComponent() 判断$_components[]中是否已存在组件实例,或$_componentConfig[]中存在组件配置信息。

public function hasComponent($id)
{
    return isset($this->_components[$id]) || isset($this->_componentConfig[$id]);
}

getComponent() 判断组件实例已经存在于$_components[]中,则直接返回对象。
否则根据$_componentConfig[]里的组件配置数据调用 Yii::createComponent() 来创建组件,并将对象存入$_components[]中然后返回。

public function getComponent($id,$createIfNull=true)
{
    if(isset($this->_components[$id]))
        return $this->_components[$id];
    else if(isset($this->_componentConfig[$id]) && $createIfNull)
    {
        $config=$this->_componentConfig[$id];
        unset($this->_componentConfig[$id]);
        if(!isset($config['enabled']) || $config['enabled'])
        {
            Yii::trace(“Loading \”$id\” application component”,’system.web.CModule’);
            unset($config['enabled']);
            $component=Yii::createComponent($config);
            $component->init();
            return $this->_components[$id]=$component;
        }
    }
}

应用组件的创建

Yii::createComponent() 来完成应用组件的创建

public static function createComponent($config)
{
    if(is_string($config))
    {
        $type=$config;
        $config=array();
    }
    else if(isset($config['class']))
    {
        $type=$config['class'];
        unset($config['class']);
    }
    else
        throw new CException(Yii::t(‘yii’,'Object configuration must be an array containing a “class” element.’));
 
    if(!class_exists($type,false))
        $type=Yii::import($type,true);
 
    if(($n=func_num_args())>1)
    {
        $args=func_get_args();
        if($n===2)
            $object=new $type($args[1]);
        else if($n===3)
            $object=new $type($args[1],$args[2]);
        else if($n===4)
            $object=new $type($args[1],$args[2],$args[3]);
        else
        {
            unset($args[0]);
            $class=new ReflectionClass($type);
            // Note: ReflectionClass::newInstanceArgs() is available for PHP 5.1.3+
            // $object=$class->newInstanceArgs($args);
            $object=call_user_func_array(array($class,’newInstance’),$args);
        }
    }
    else
        $object=new $type;
 
    foreach($config as $key=>$value)
    $object->$key=$value;
 
    return $object;
}

 以上就是Yii框架分析(三)——类加载机制及应用组件的管理、配置、访问、创建的内容,更多相关内容请关注PHP中文网(www.php.cn)!


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.