《PHP實戰(zhàn):深入講解PHP的Yii框架中的屬性(Property)》要點:
本文介紹了PHP實戰(zhàn):深入講解PHP的Yii框架中的屬性(Property),希望對您有用。如果有疑問,可以聯(lián)系我們。
相關主題:YII框架
PHP教程在 PHP 中,類的成員變量也被稱為屬性(properties).它們是類定義的一部分,用來表現(xiàn)一個實例的狀態(tài)(也就是區(qū)分類的不同實例).在具體實踐中,常常會想用一個稍微特殊些的辦法實現(xiàn)屬性的讀寫.例如,如果有需求每次都要對 label 屬性執(zhí)行 trim 操作,就可以用以下代碼實現(xiàn):
PHP教程
$object->label = trim($label);
PHP教程上述代碼的缺點是只要修改 label 屬性就必須再次調(diào)用 trim() 函數(shù).若將來需要用其它方式處理 label 屬性,好比首字母大寫,就不得不修改所有給 label 屬性賦值的代碼.這種代碼的重復會導致 bug,這種實踐顯然需要盡可能避免.
PHP教程為解決該問題,Yii 引入了一個名為 yii\base\Object 的基類,它支持基于類內(nèi)的 getter 和 setter(讀取器和設定器)辦法來定義屬性.如果某類需要支持這個特性,只需要繼承 yii\base\Object 或其子類即可.
PHP教程補充:幾乎每個 Yii 框架的核心類都繼承自 yii\base\Object 或其子類.這意味著只要在核心類中見到 getter 或 setter 辦法,就可以像調(diào)用屬性一樣調(diào)用它.
getter 辦法是名稱以 get 開頭的辦法,而 setter 辦法名以 set 開頭.辦法名中 get 或 set 后面的部分就定義了該屬性的名字.如下面代碼所示,getter 辦法 getLabel() 和 setter 辦法 setLabel() 操作的是 label 屬性,:
PHP教程
namespace app\components;
use yii\base\Object;
class Foo extend Object
{
private $_label;
public function getLabel()
{
return $this->_label;
}
public function setLabel($value)
{
$this->_label = trim($value);
}
}
PHP教程(詳細解釋:getter 和 setter 辦法創(chuàng)建了一個名為 label 的屬性,在這個例子里,它指向一個私有的內(nèi)部屬性 _label.)
PHP教程getter/setter 定義的屬性用法與類成員變量一樣.兩者主要的區(qū)別是:當這種屬性被讀取時,對應的 getter 辦法將被調(diào)用;而當屬性被賦值時,對應的 setter 辦法就調(diào)用.如:
PHP教程
// 等效于 $label = $object->getLabel();
$label = $object->label;
// 等效于 $object->setLabel('abc');
$object->label = 'abc';
PHP教程只定義了 getter 沒有 setter 的屬性是只讀屬性.嘗試賦值給這樣的屬性將導致 yii\base\InvalidCallException (無效調(diào)用)異常.類似的,只有 setter 辦法而沒有 getter 辦法定義的屬性是只寫屬性,嘗試讀取這種屬性也會觸發(fā)異常.使用只寫屬性的情況幾乎沒有.
PHP教程通過 getter 和 setter 定義的屬性也有一些特殊規(guī)則和限制:
PHP教程這類屬性的名字是不區(qū)分大小寫的.如,$object->label 和 $object->Label 是同一個屬性.因為 PHP 方法名是不區(qū)分大小寫的.
如果此類屬性名和類成員變量相同,以后者為準.例如,假設以上 Foo 類有個 label 成員變量,然后給 $object->label = 'abc' 賦值,將賦給成員變量而不是 setter setLabel() 方法.
這類屬性不支持可見性(拜訪限制).定義屬性的 getter 和 setter 方法是 public、protected 還是 private 對屬性的可見性沒有任何影響.
這類屬性的 getter 和 setter 方法只能定義為非靜態(tài)的,若定義為靜態(tài)方法(static)則不會以相同方式處理.
回到開頭提到的問題,與其處處要調(diào)用 trim() 函數(shù),現(xiàn)在我們只需在 setter setLabel() 方法內(nèi)調(diào)用一次.如果 label 首字母變成大寫的新要求來了,我們只需要修改setLabel() 方法,而無須接觸任何其它代碼.
PHP教程實現(xiàn)屬性的步驟
PHP教程我們知道,在讀取和寫入對象的一個不存在的成員變量時, __get() __set() 會被自動調(diào)用. Yii正是利用這點,提供對屬性的支持的.從上面的代碼中,可以看出,如果拜訪一個對象的某個屬性, Yii會調(diào)用名為 get屬性名() 的函數(shù).如, SomeObject->Foo , 會自動調(diào)用 SomeObject->getFoo() .如果修改某一屬性,會調(diào)用相應的setter函數(shù). 如, SomeObject->Foo = $someValue ,會自動調(diào)用 SomeObject->setFoo($someValue) .
PHP教程因此,要實現(xiàn)屬性,通常有三個步驟:
PHP教程如下的Post類,實現(xiàn)了可讀可寫的屬性title:
PHP教程
class Post extends yii\base\Object // 第一步:繼承自 yii\base\Object
{
private $_title; // 第二步:聲明一個私有成員變量
public function getTitle() // 第三步:提供getter和setter
{
return $this->_title;
}
public function setTitle($value)
{
$this->_title = trim($value);
}
}
PHP教程從理論上來講,將 private $_title 寫成 public $title ,也是可以實現(xiàn)對 $post->title 的讀寫的.但這不是好的習慣,理由如下:
PHP教程失去了類的封裝性. 一般而言,成員變量對外不可見是比較好的編程習慣. 從這里你也許沒看出來,但是假如有一天,你不想讓用戶修改標題了,你怎么改? 怎么確保代碼中沒有直接修改標題? 如果提供了setter,只要把setter刪掉,那么一旦有沒清理干凈的對標題的寫入,就會拋出異常. 而使用 public $title 的辦法的話,你改成 private $title 可以排查寫入的異常,但是讀取的也被禁止了.
對于標題的寫入,你想去掉空格. 使用setter的辦法,只需要像上面的代碼段一樣在這個地方調(diào)用 trim() 就可以了. 但如果使用 public $title 的辦法,那么毫無疑問,每個寫入語句都要調(diào)用 trim() . 你能保證沒有一處遺漏?
因此,使用 public $title 只是一時之快,看起來簡單,但今后的修改是個麻煩事. 簡直可以說是惡夢.這就是軟件工程的意義所在,通過一定的辦法,使代碼易于維護、便于修改. 一時看著好像沒必要,但實際上吃過虧的朋友或者被客戶老板逼著修改上一個程序員寫的代碼,問候過他親人的, 都會覺得這是十分必要的.
PHP教程但是,世事無絕對.由于 __get() 和 __set() 是在遍歷所有成員變量,找不到匹配的成員變量時才被調(diào)用. 因此,其效率天生地低于使用成員變量的形式.在一些表示數(shù)據(jù)結(jié)構、數(shù)據(jù)集合等簡單情況下,且不需讀寫控制等, 可以考慮使用成員變量作為屬性,這樣可以提高一點效率.
PHP教程另外一個提高效率的小技巧就是:使用 $pro = $object->getPro() 來代替 $pro = $object->pro , 用 $objcect->setPro($value) 來代替 $object->pro = $value . 這在功能上是完全一樣的效果,但是避免了使用 __get() 和 __set() ,相當于繞過了遍歷的過程.
PHP教程這里估計有人該罵我了,Yii好不容易實現(xiàn)了屬性的機制,就是為了方便開發(fā)者, 結(jié)果我卻在這里教大家怎么使用原始的方式,去提高所謂的效率. 嗯,確實,開發(fā)的便利性與執(zhí)行高效率存在一定的矛盾.我個人的觀點更傾向于以便利為先, 用好、用足Yii為我們創(chuàng)造的便利條件.至于效率的事情,更多的是框架自身需要注意的, 我們只要別寫出格外2的代碼就OK了.
PHP教程不過你完全可以放心,在Yii的框架中,極少出現(xiàn) $app->request 之類的代碼,而是使用 $app->getRequest() . 換句話說,框架自身還是格外地注重效率的,至于便利性,則留給了開發(fā)者. 總之,這里只是點出來有這么一個知識點,至于用不用,怎么用,完全取決于你了.
PHP教程值得注意的是:
PHP教程由于自動調(diào)用 __get() __set() 的時機僅僅發(fā)生在拜訪不存在的成員變量時. 因此,如果定義了成員變量 public $title 那么,就算定義了 getTitle() setTitle() , 他們也不會被調(diào)用.因為 $post->title 時,會直接指向該 pulic $title , __get() __set() 是不會被調(diào)用的.從根上就被切斷了.
由于PHP對于類方法不區(qū)分大小寫,即大小寫不敏感, $post->getTitle() 和 $post->gettitle() 是調(diào)用相同的函數(shù). 因此, $post->title 和 $post->Title 是同一個屬性.即屬性名也是不區(qū)分大小寫的.
由于 __get() __set() 都是public的, 無論將 getTitle() setTitle() 聲明為 public, private, protected, 都沒有意義,外部同樣都是可以拜訪.所以,所有的屬性都是public的.
由于 __get() __set() 都不是static的,因此,沒有辦法使用static 的屬性.
Object的其他與屬性相關的方法
PHP教程除了 __get() __set() 之外, yii\base\Object 還提供了以下辦法便于使用屬性:
PHP教程yii\base\Component 繼承自 yii\base\Object ,因此,他也具有屬性等基本功能.
PHP教程但是,由于Componet還引入了事件、行為,因此,它并非簡單繼承了Object的屬性實現(xiàn)方式,而是基于同樣的機制, 重載了 __get() __set() 等函數(shù).但從實現(xiàn)機制上來講,是一樣的.這個不影響理解.
PHP教程前面說過,官方將Yii定位于一個基于組件的框架.可見組件這一概念是Yii的基礎. 如果你有興趣閱讀Yii的源代碼或是API文檔,你將會發(fā)現(xiàn), Yii幾乎所有的核心類都派生于(繼承自) yii\base\Component .
PHP教程在Yii1.1時,就已經(jīng)有了component了,那時是 CComponent.Yii2將Yii1.1中的CComponent拆分成兩個類: yii\base\Object 和 yii\base\Component .
PHP教程其中,Object比擬輕量級些,通過getter和setter定義了類的屬性(property). Component派生自Object,并支持事件(event)和行為(behavior).因此,Component類具有三個重要的特性:
PHP教程相信你或多或少了解過,這三個特性是豐富和拓展類功能、改變類行為的重要切入點. 因此,Component在Yii中的地位極高.
PHP教程在提供更多功能、更多便利的同時,Component由于增加了event和behavior這兩個特性, 在方便開發(fā)的同時,也犧牲了一定的效率. 如果開發(fā)中不需要使用event和behavior這兩個特性,好比表示一些數(shù)據(jù)的類. 那么,可以不從Component繼承,而從Object繼承. 典型的應用場景就是如果表示用戶輸入的一組數(shù)據(jù),那么,使用Object. 而如果需要對對象的行為和能響應處理的事件進行處理,毫無疑問應當采用Component. 從效率來講,Object更接近原生的PHP類,因此,在可能的情況下,應當優(yōu)先使用Object.
維易PHP培訓學院每天發(fā)布《PHP實戰(zhàn):深入講解PHP的Yii框架中的屬性(Property)》等實戰(zhàn)技能,PHP、MYSQL、LINUX、APP、JS,CSS全面培養(yǎng)人才。
轉(zhuǎn)載請注明本頁網(wǎng)址:
http://www.snjht.com/jiaocheng/7305.html