PHP靜態(tài)綁定與動(dòng)態(tài)綁定
除了限制訪問(wèn),訪問(wèn)方式也決定哪個(gè)方法將被子類(lèi)調(diào)用或哪個(gè)屬性將被子類(lèi)訪問(wèn). 函數(shù)調(diào)用與函數(shù)本身的關(guān)聯(lián),以及成員訪問(wèn)與變量?jī)?nèi)存地址間的關(guān)系,稱(chēng)為綁定.
在計(jì)算機(jī)語(yǔ)言中有兩種主要的綁定方式?靜態(tài)綁定和動(dòng)態(tài)綁定. 靜態(tài)綁定發(fā)生于數(shù)據(jù)結(jié)構(gòu)和數(shù)據(jù)結(jié)構(gòu)間,程序執(zhí)行之前. 靜態(tài)綁定發(fā)生于編譯期, 因此不能利用任何運(yùn)行期的信息. 它針對(duì)函數(shù)調(diào)用與函數(shù)的主體,或變量與內(nèi)存中的區(qū)塊. 因?yàn)镻HP是一種動(dòng)態(tài)語(yǔ)言,它不使用靜態(tài)綁定. 但是可以模擬靜態(tài)綁定.
動(dòng)態(tài)綁定則針對(duì)運(yùn)行期產(chǎn)生的訪問(wèn)請(qǐng)求,只用到運(yùn)行期的可用信息. 在面向?qū)ο蟮拇a中,動(dòng)態(tài)綁定意味著決定哪個(gè)方法被調(diào)用或哪個(gè)屬性被訪問(wèn),將基于這個(gè)類(lèi)本身而不基于訪問(wèn)范圍.
Public和protected成員的動(dòng)作類(lèi)似于PHP的前幾個(gè)版本中函數(shù)的動(dòng)作,使用動(dòng)態(tài)綁定. 這意味著如果一個(gè)方法訪問(wèn)一個(gè)在子類(lèi)中被覆寫(xiě)的類(lèi)成員,并是一個(gè)子類(lèi)的實(shí)例,子類(lèi)的成員將被訪問(wèn)(而不是訪問(wèn)父類(lèi)中的成員).
看圖1. 這段代碼輸出” Hey! I am Son.” 因?yàn)楫?dāng)PHP調(diào)用getSalutation, 是一個(gè)Son的實(shí)例,是將Father中的salutation覆寫(xiě)而來(lái). 如果salutation是public的,PHP將產(chǎn)生相同的結(jié)果. 覆寫(xiě)方法的操作很類(lèi)似.在Son中,對(duì)于identify的調(diào)用綁定到那個(gè)方法.
即使在子類(lèi)中訪問(wèn)方式被從protected削弱成public, 動(dòng)態(tài)綁定仍然會(huì)發(fā)生. 按照訪問(wèn)方式使用的原則,增強(qiáng)對(duì)于類(lèi)成員的訪問(wèn)限制是不可能的. 所以把訪問(wèn)方式從public改變成protected不可能進(jìn)行.
動(dòng)態(tài)綁定
<?php class Father { protected $salutation = 'Hello there!'; //問(wèn)候
public function getSalutation() { print('$this->salutationn'); $this->identify(); }
protected function identify() { print('I am Father.n'); } };
class Son extends Father { protected $salutation = 'Hey!'//父類(lèi)中的protected $salutation 被覆寫(xiě)
protected function identify();//父類(lèi)中的protected identify() 被覆寫(xiě) { print('I am Son.n'); } };
$obj = new Son(); $obj->getSalutation()//輸出Hey! I am Son. ?>
//注: 在子類(lèi)中沒(méi)有覆寫(xiě)getSalutation(),但實(shí)際上仍然存在一getSalutation().這個(gè)類(lèi)中的$salutation和identify()
//與Son子類(lèi)的實(shí)例中的getSalutation()方法動(dòng)態(tài)綁定,所以調(diào)用Son的實(shí)例的getSalutation()方法,
//將調(diào)用Son類(lèi)中的成員salutation及identify(),而不是父類(lèi)中的成員salutation及identify().
Private成員只存在于它們所在的類(lèi)內(nèi)部. 不像public和protected成員那樣,PHP模擬靜態(tài)綁定. 看例子圖2 . 它輸出”Hello there! I am Father.”,盡管子類(lèi)覆寫(xiě)了salutation的值. 腳本將this->salutation和當(dāng)前類(lèi)Father綁定. 類(lèi)似的原則應(yīng)用于private方法identify().
Binding and private members
<?php class Father { private $salutation = 'Hello there!';
public function getSalutation() { print('$this->salutationn'); $this->identify(); }
private function identify() { print('I am Father.n'); } }
class Son extends Father { private $salutation = 'Hey!'; private function identify() { print('I am Son.n'); } }
$obj = new Son(); $obj->getSalutation(); //輸出Hello there! I am Father. ?>
動(dòng)態(tài)綁定的好處是允許繼承類(lèi)來(lái)改變父類(lèi)的行為,同時(shí)可以保持父類(lèi)的接口和功能. 看例子圖3. 由于使用了動(dòng)態(tài)綁定,在deleteUser中被調(diào)用的isAuthorized的version 可以由對(duì)象的類(lèi)型來(lái)確定. 如果是一個(gè)普通的user,PHP調(diào)用User::isAuthorized會(huì)返回FALSE.如果是一個(gè)AuthorizedUser的實(shí)例,PHP調(diào)用AuthorizedUser::isAuthorized,將允許deleteUser順利執(zhí)行.
//haohappy注:用一句話(huà)說(shuō)清楚,就是對(duì)象類(lèi)型與方法,屬性綁定. 調(diào)用一個(gè)父類(lèi)與子類(lèi)中都存在的方法或訪問(wèn)一個(gè)屬性時(shí),會(huì)先判斷實(shí)例屬于哪種對(duì)象類(lèi)型,再調(diào)用相應(yīng)的類(lèi)中的方法和屬性.
動(dòng)態(tài)綁定的好處
<?php class User //用戶(hù) { protected function isAuthorized(); //是否是驗(yàn)證用戶(hù) { return(FALSE); }
public function getName(); //獲得名字 { return($this->name); }
public function deleteUser($username) //刪除用戶(hù) { if(!$this->isAuthorized()) { print('You are not authorized.n'); return(FALSE); }
//delete the user print('User deleted.n'); } }
class AuthorizedUser extends User //認(rèn)證用戶(hù) { protected function isAuthorized() //覆寫(xiě)isAuthorized() { return(TRUE); } }
$user = new User; $admin = new AuthorizedUser;
//not authorized $user->deleteUser('Zeev');
//authorized $admin->deleteUser('Zeev'); ?>
為什么private的類(lèi)成員模擬靜態(tài)綁定? 為了回答這個(gè)問(wèn)題, 你需要回憶一下為什么需要有private成員.什么時(shí)候用它們來(lái)代替protected成員是有意義的?
private成員只有當(dāng)你不想讓子類(lèi)繼承改變或特殊化父類(lèi)的行為時(shí)才用到. 這種情況比你想像的要少. 通常來(lái)說(shuō),一個(gè)好的對(duì)象分層結(jié)構(gòu)應(yīng)當(dāng)允許絕大多數(shù)功能被子類(lèi)特殊化,改進(jìn),或改變?這是面向?qū)ο缶幊痰幕A(chǔ)之一. 一定的情況下需要private方法或變量,例如當(dāng)你確信你不想允許子類(lèi)改變父類(lèi)中的某個(gè)特定的部份.
