明輝手游網(wǎng)中心:是一個(gè)免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺(tái)!

第9節(jié) 綁定 [9]

[摘要]除了限制訪問,訪問方式也決定哪個(gè)方法將被子類調(diào)用或哪個(gè)屬性將被子類訪問. 函數(shù)調(diào)用與函數(shù)本身的關(guān)聯(lián),以及成員訪問與變量內(nèi)存地址間的關(guān)系,稱為綁定. 在計(jì)算機(jī)語言中有兩種主要的綁定方式—靜態(tài)綁定和動(dòng)態(tài)綁定. 靜態(tài)綁定發(fā)生于數(shù)據(jù)結(jié)構(gòu)和數(shù)據(jù)結(jié)構(gòu)間,程序執(zhí)行之前. 靜態(tài)綁定發(fā)生于編譯期, 因此不能利用任何...

除了限制訪問,訪問方式也決定哪個(gè)方法將被子類調(diào)用或哪個(gè)屬性將被子類訪問. 函數(shù)調(diào)用與函數(shù)本身的關(guān)聯(lián),以及成員訪問與變量內(nèi)存地址間的關(guān)系,稱為綁定.

在計(jì)算機(jī)語言中有兩種主要的綁定方式—靜態(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)語言,它不使用靜態(tài)綁定. 但是可以模擬靜態(tài)綁定.

動(dòng)態(tài)綁定則針對(duì)運(yùn)行期產(chǎn)生的訪問請(qǐng)求,只用到運(yùn)行期的可用信息. 在面向?qū)ο蟮拇a中,動(dòng)態(tài)綁定意味著決定哪個(gè)方法被調(diào)用或哪個(gè)屬性被訪問,將基于這個(gè)類本身而不基于訪問范圍.

Public和protected成員的動(dòng)作類似于PHP的前幾個(gè)版本中函數(shù)的動(dòng)作,使用動(dòng)態(tài)綁定. 這意味著如果一個(gè)方法訪問一個(gè)在子類中被覆寫的類成員,并是一個(gè)子類的實(shí)例,子類的成員將被訪問(而不是訪問父類中的成員).

看例子6.10. 這段代碼輸出” Hey! I am Son.” 因?yàn)楫?dāng)PHP調(diào)用getSalutation, 是一個(gè)Son的實(shí)例,是將Father中的salutation覆寫而來. 如果salutation是public的,PHP將產(chǎn)生相同的結(jié)果. 覆寫方法的操作很類似.在Son中,對(duì)于identify的調(diào)用綁定到那個(gè)方法.

即使在子類中訪問方式被從protected削弱成public, 動(dòng)態(tài)綁定仍然會(huì)發(fā)生. 按照訪問方式使用的原則,增強(qiáng)對(duì)于類成員的訪問限制是不可能的. 所以把訪問方式從public改變成protected不可能進(jìn)行.

Listing 6.10 Dynamic binding 動(dòng)態(tài)綁定
<?php
class Father
{
protected $salutation = "Hello there!"; file://問候

public function getSalutation()
{
print("$this->salutationn");
$this->identify();
}

protected function identify()
{
print("I am Father.<br>n");
}
};

class Son extends Father
{
protected $salutation = "Hey!"; file://父類中的protected $salutation 被覆寫

protected function identify() file://父類中的protected identify() 被覆寫
{
print("I am Son.<br>n");
}
};

$obj = new Son();
$obj->getSalutation(); file://輸出Hey! I am Son.
?> //注: 在子類中沒有覆寫getSalutation(),但實(shí)際上仍然存在一個(gè)getSalutation().這個(gè)類中的$salutation和identify()
//與Son子類的實(shí)例中的getSalutation()方法動(dòng)態(tài)綁定,所以調(diào)用Son的實(shí)例的getSalutation()方法,
//將調(diào)用Son類中的成員salutation及identify(),而不是父類中的成員salutation及identify().

Private成員只存在于它們所在的類內(nèi)部. 不像public和protected成員那樣,PHP模擬靜態(tài)綁定. 看例子6.11. 它輸出”Hello there! I am Father.”,盡管子類覆寫了salutation的值. 腳本將this->salutation和當(dāng)前類Father綁定. 類似的原則應(yīng)用于private方法identify().

Listing 6.11 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.<br>n");
}
}

class Son extends Father
{
private $salutation = "Hey!";
private function identify()
{
print("I am Son.<br>n");
}
}

$obj = new Son();
$obj->getSalutation(); file://輸出Hello there! I am Father.
?>動(dòng)態(tài)綁定的好處是允許繼承類來改變父類的行為,同時(shí)可以保持父類的接口和功能. 看例子6.12. 由于使用了動(dòng)態(tài)綁定,在deleteUser中被調(diào)用的isAuthorized的version 可以由對(duì)象的類型來確定. 如果是一個(gè)普通的user,PHP調(diào)用User::isAuthorized會(huì)返回FALSE.如果是一個(gè)AuthorizedUser的實(shí)例,PHP調(diào)用AuthorizedUser::isAuthorized,將允許deleteUser順利執(zhí)行.

//haohappy注:用一句話說清楚,就是對(duì)象類型與方法,屬性綁定. 調(diào)用一個(gè)父類與子類中都存在的方法或訪問一個(gè)屬性時(shí),會(huì)先判斷實(shí)例屬于哪種對(duì)象類型,再調(diào)用相應(yīng)的類中的方法和屬性.

Listing 6.12 動(dòng)態(tài)綁定的好處
<?php
class User file://用戶
{
protected function isAuthorized() file://是否是驗(yàn)證用戶
{
return(FALSE);
}

public function getName() file://獲得名字
{
return($this->name);
}

public function deleteUser($username) file://刪除用戶
{
if(!$this->isAuthorized())
{
print("You are not authorized.<br>n");
return(FALSE);
}

//delete the user
print("User deleted.<br>n");
}
}

class AuthorizedUser extends User file://認(rèn)證用戶
{
protected function isAuthorized() file://覆寫isAuthorized()
{
return(TRUE);
}
}

$user = new User;
$admin = new AuthorizedUser;

//not authorized
$user->deleteUser("Zeev");

//authorized
$admin->deleteUser("Zeev");
?> 為什么private的類成員模擬靜態(tài)綁定? 為了回答這個(gè)問題, 你需要回憶一下為什么需要有private成員.什么時(shí)候用它們來代替protected成員是有意義的?

private成員只有當(dāng)你不想讓子類繼承改變或特殊化父類的行為時(shí)才用到. 這種情況比你想像的要少. 通常來說,一個(gè)好的對(duì)象分層結(jié)構(gòu)應(yīng)當(dāng)允許絕大多數(shù)功能被子類特殊化,改進(jìn),或改變—這是面向?qū)ο缶幊痰幕A(chǔ)之一. 一定的情況下需要private方法或變量,例如當(dāng)你確信你不想允許子類改變父類中的某個(gè)特定的部份.


標(biāo)簽:第9節(jié) 綁定 [9]