구성 요소는 Yii 애플리케이션의 주요 구성 요소입니다. yiibaseComponent 클래스 또는 해당 하위 클래스의 인스턴스입니다. 다른 클래스와 구별하는 데 사용되는 세 가지 주요 기능은 다음과 같습니다.
단독으로 사용하거나 서로 결합하여 사용하면 Yii 클래스를 더욱 유연하고 쉽게 사용할 수 있습니다. 위젯 yiijuiDatePicker를 예로 들어보겠습니다. 이는 보기에서 대화형 날짜 선택기를 생성할 수 있는 UI 구성 요소입니다.
use yii\jui\DatePicker; echo DatePicker::widget([ 'language' => 'zh-CN', 'name' => 'country', 'clientOptions' => [ 'dateFormat' => 'yy-mm-dd', ], ]);
이 위젯은 yiibaseComponent에서 상속되며 해당 속성을 쉽게 다시 작성할 수 있습니다.
정확하게는 구성 요소의 강력한 기능 때문에 이벤트와 동작을 처리하는 데 추가 메모리와 CPU 시간을 사용하기 때문에 일반 개체(Object)보다 약간 무겁습니다. 이 두 가지 기능이 필요하지 않은 경우 yiibaseComponent 대신 yiibaseObject를 상속할 수 있습니다. 이러한 방식으로 구성 요소는 일반 PHP 개체만큼 효율적이면서 속성 기능도 지원할 수 있습니다.
yiibaseComponent 또는 yiibaseObject를 상속할 때 다음 코딩 스타일을 사용하는 것이 좋습니다.
생성자(Constructor)를 재정의해야 하는 경우 생성자 메서드의 마지막 매개변수로 $config를 전달한 후 상위 클래스의 생성자에 전달합니다.
항상 재정의된 생성자의 끝에서 부모 클래스의 생성자를 호출하세요.
yiibaseObject::init() 메서드를 재정의하는 경우 init 메서드 시작 부분에서 부모 클래스의 init 메서드를 호출해야 합니다.
예시는 다음과 같습니다.
namespace yii\components\MyClass; use yii\base\Object; class MyClass extends Object { public $prop1; public $prop2; public function __construct($param1, $param2, $config = []) { // ... 配置生效前的初始化过程 parent::__construct($config); } public function init() { parent::init(); // ... 配置生效后的初始化过程 } }
또한 인스턴스 생성 시 구성 요소가 올바르게 구성되도록 하려면 다음 작업 프로세스를 따르십시오.
$component = new MyClass(1, 2, ['prop1' => 3, 'prop2' => 4]); // 方法二: $component = \Yii::createObject([ 'class' => MyClass::className(), 'prop1' => 3, 'prop2' => 4, ], [1, 2]);
보충: Yii::createObject()를 호출하는 메서드가 더 복잡해 보이지만 이는 더 유연하고 강력하며 종속성 주입 컨테이너를 기반으로 구현되기 때문입니다.
실행 시 yiibaseObject 클래스의 라이프사이클은 다음과 같습니다.
생성자 내의 사전 초기화 프로세스. 여기에서 각 속성에 대한 기본값을 설정할 수 있습니다.
$config를 통한 구성 개체. 구성 프로세스는 이전에 생성자에서 설정한 기본값을 덮어쓸 수 있습니다.
yiibaseObject::init() 메소드에서 초기화 후 마무리 작업을 수행합니다. 이 메서드를 재정의하여 품질 검사 및 속성 초기화와 같은 일부 작업을 수행할 수 있습니다.
객체 메소드 호출.
처음 세 단계는 모두 객체 생성자 내에서 발생합니다. 즉, 객체 인스턴스를 얻으면 초기화되어 사용할 준비가 됩니다.
애플리케이션 CWebApplication 구성 요소
Yii의 각 컴포넌트 사용법을 설명하기에 앞서 가장 중요한 컴포넌트인 CWebApplication에 대해 먼저 알아보겠습니다. CWebApplication은 애플리케이션 객체이고, 그 루트 클래스도 CComponent이므로 역시 컴포넌트이고 Yii 컴포넌트의 공통적인 특징을 가지고 있습니다.
구체적으로 CWebApplication 구성 요소의 주요 기능은 구성 파일에 따라 필요한 보조 구성 요소를 로드하고 이러한 구성 요소(예: urlManager)의 도움으로 컨트롤러를 생성 및 실행하는 것입니다. 따라서 프론트 엔드 컨트롤러라고도 합니다.
구성 파일에서 CWebApplication 구성 요소 자체의 구성 매개 변수를 지정할 수 있습니다. 이러한 매개 변수는 공용 멤버 변수로 설정되거나 setter 메서드가 자동으로 호출되어 속성을 설정합니다. 이 기능은 CWebApplication 생성자에서 찾을 수 있습니다. ->구성($config);
protected/config/main.php 구성 파일에 전역적으로 지정된 대로:
'charset' => 'utf-8',
这实际是设置当前应用程序的charset公共属性(在CApplication中声明)而如果在配置文件中指定'language' => 'zh_cn', 我们发现CWebApplication及其所有上级类均未声明$language属性,这时将使用setter模式方法即setlanuage(此方法定义在CApplication类中)。
OK,了解这个特性之后,我们就可以明白在配置文件中可以配置的属性:
CWebApplication的继承层次为:CApplication -> CModule -> CComponent, 我们将默认的配置文件中常见的配置项及其生效位置予以说明:
类似地,再列出几个默认配置文件中并未列出的配置项:timezone: CApplication::setTimeZone() #配置时区
再例如,如果我们继承CWebApplication, 扩展自己的应用程序类myApp, 并定义方法setError_reporting(不区分大小写), 那么就可以直接在配置文件中指定error_reporting选项。
辅助组件可以将CWebApplication组件视为一部机器,那么辅助组件就可以视为组成这部机器的各个零件,没有零件的正确组合,机器就无法正常工作,这在Yii中也是同样的概念。而一些组件对整部机器的运转是必须的,这就是核心组件。在应用程序对象构造后,Yii会将辅助组件基本信息进行登记(组件名称与类名,属性配置的对照表),以供后续使用,对web应用程序而言,存在以下核心组件(通过CWebApplication::registerCoreComponents,CApplication::registerCoreComponents注册):
CWebApplication::registerCoreComponents中注册的核心组件
CApplication::registerCoreComponents中注册的核心组件
配置文本中注册的核心组件:log CLogRouter 日志路由管理器
以上标记为红色的条目,是最重要的辅助组件,其它的核心组件我们未必会使用到。
如何定义辅助组件的属性?通过在配置文件protected/config/main.php中设置components项的值,实现组件属性定义。这里的定义主要是三个要素:指定组件的名称(核心组件已经预先设置)、指定组件使用的类(核心组件无须定义),组件的属性(可选、视情况而定)
如以下配置:
'components' => array( 'db' => array( 'class' => 'myCDbConnection', 'connnectionString' => 'mysql:host=localhost;dbname=test;charset=utf8', 'user' => 'root', ), );
就设置了db组件使用的类为myCDbConnection, 并且在后面指定了连接串及账号等信息。提示: myCDbConnection类可能就是通过继承CDbConnection类定义。核心组件无须指定class参数(因为已经预先定义好)
问题:如何得知某个组件可配置的属性?这个问题至关重要,如果我们掌握了规律,就可以举一反三,所有组件的配置均可以灵活设定。授之以鱼不如授之以渔。在本节会说明通用的方法。要得知组件的所有可定义属性,按以下步骤进行:
1. 组件所使用的类是什么?(无论是核心组件还是自定义组件)
2. 组件类的公共成员变量都有哪些?(注意从父类继承而来的公共成员变量)
3. 组件类都有哪些settter方法?(注意从父类继承而来的方法)
明白了以上三个要点,我们就可以按规律定义组件的属性,比如对最重要的db组件,我们发现这是一个核心组件,使用的类为CDbConnection, 我们查阅这个类的定义文件,发现这个类的公共成员变量有:
$connectionString;
setter方法定义的属性:
提示:setter方法定义的属性名称不区分大小写以上所列的属性,均可以在配置文件中指定,具体每个属性的作用,请参阅Yii类文件的详细注释(Yii代码的注释也是相当棒,通俗易懂,而又很详细)
再来一个例子,定义urlManager组件的属性这个组件使用的类为CUrlManager, 我们查阅它的属性:
通过setter方法定义的属性:
即urlManager组件的上述属性可以在配置文件中定义(每项配置的作用请参阅其注释)。其它组件的配置均可按上述方法处理。
如何使用组件应用程序运行后,会将所有已经定义过的组件注册(并未实例化)到CWebApplication对象上,同时CWebApplication应用程序对象会被注册到Yii::$_app,在程序的任何位置均可通过Yii::app()得到当前应用程序对象引用,再通过$app对象得到组件实例引用,如:Yii::app()->getComponent('urlManager'); #会查找组件配置并实例化之Yii::app()->urlManager; #通过CModule::__get()魔术方法实现
如何自定义组件?这是很常见的需求,比如我们可能希望db组件(数据库连接)使用我们自定义的类,也或者我们希望使用多个数据库连接,这种情况下就需要自定义组件,使用多数据库的例子:
components=>array( 'db' => array( ... ... ), 'mydb'=>array( 'class' => 'myDbConnection', 'connectionString' => 'mysql:host=localhost;dbname=test;charset=utf8', 'tablePrefix' => 'cdb_', 'username' => 'root', ), ), 修改默认的db组件所使用的类: components=>array( 'db' => array( 'class' => 'myDbConnection', ... ... ), ),
经过本文的分析,我是深切理解了Yii组件化机制给应用程序带来的极大的扩展性,哈哈哈哈~