>백엔드 개발 >PHP 튜토리얼 >PHP Yii Framework_php 기술의 구성 요소화 메커니즘에 대한 기본 지식을 간략하게 분석합니다.

PHP Yii Framework_php 기술의 구성 요소화 메커니즘에 대한 기본 지식을 간략하게 분석합니다.

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB원래의
2016-05-16 19:56:371064검색

구성 요소는 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及其所有上级类的公共成员变量
  • CWebApplication及其所有上级类的setter方法指定的属性当然我们也可以通过继承CWebApplication构造自己的应用程序类。

CWebApplication的继承层次为:CApplication -> CModule -> CComponent, 我们将默认的配置文件中常见的配置项及其生效位置予以说明:

  • basePath :  CApplication::setBasePath()
  • name: CApplication::$name
  • preload: CModule::$preload
  • import: CModule::setImport()
  • defaultController: CWebApplication::$defaultController
  • components: CModule::setComponents()

类似地,再列出几个默认配置文件中并未列出的配置项:timezone: CApplication::setTimeZone()  #配置时区

再例如,如果我们继承CWebApplication, 扩展自己的应用程序类myApp, 并定义方法setError_reporting(不区分大小写), 那么就可以直接在配置文件中指定error_reporting选项。
辅助组件可以将CWebApplication组件视为一部机器,那么辅助组件就可以视为组成这部机器的各个零件,没有零件的正确组合,机器就无法正常工作,这在Yii中也是同样的概念。而一些组件对整部机器的运转是必须的,这就是核心组件。在应用程序对象构造后,Yii会将辅助组件基本信息进行登记(组件名称与类名,属性配置的对照表),以供后续使用,对web应用程序而言,存在以下核心组件(通过CWebApplication::registerCoreComponents,CApplication::registerCoreComponents注册):

CWebApplication::registerCoreComponents中注册的核心组件

2016317152903914.png (629×145)

CApplication::registerCoreComponents中注册的核心组件

2016317152921101.png (645×160)

配置文本中注册的核心组件: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;

  • $username='';
  • $password='';
  • $autoConnect=true;
  • $charset;
  • $emulatePrepare;
  • $tablePrefix;
  • $initSQLs;
  • ... ...

setter方法定义的属性:

  • setActive($value)
  • setAttributes($values)
  • setAutoCommit($value)
  • setColumnCase($value)
  • setNullConversion($value)
  • setPersistent($value)

提示:setter方法定义的属性名称不区分大小写以上所列的属性,均可以在配置文件中指定,具体每个属性的作用,请参阅Yii类文件的详细注释(Yii代码的注释也是相当棒,通俗易懂,而又很详细)

再来一个例子,定义urlManager组件的属性这个组件使用的类为CUrlManager, 我们查阅它的属性:

  • $rules=array();
  • $urlSuffix='';
  • $showScriptName=true;
  • $appendParams=true;
  • $routeVar='r';
  • $caseSensitive=true;

通过setter方法定义的属性:

  • setUrlFormat($value)
  • setBaseUrl($value)

即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组件化机制给应用程序带来的极大的扩展性,哈哈哈哈~

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