PHP靜態繫結與動態綁定_PHP教程

來源:互聯網
上載者:User

除了限制訪問,訪問方式也決定哪個方法將被子類調用或哪個屬性將被子類訪問. 函數調用與函數本身的關聯,以及成員訪問與變數記憶體位址間的關係,稱為綁定.

在電腦語言中有兩種主要的綁定方式?靜態繫結和動態綁定. 靜態繫結發生於資料結構和資料結構間,程式執行之前. 靜態繫結發生於編譯期, 因此不能利用任何運行期的資訊. 它針對函數調用與函數的主體,或變數與記憶體中的區塊. 因為PHP是一種動態語言,它不使用靜態繫結. 但是可以類比靜態繫結.

動態綁定則針對運行期產生的訪問請求,只用到運行期的可用資訊. 在物件導向的代碼中,動態綁定意味著決定哪個方法被調用或哪個屬性被訪問,將基於這個類本身而不基於存取範圍.

Public和protected成員的動作類似於PHP的前幾個版本中函數的動作,使用動態綁定. 這意味著如果一個方法訪問一個在子類中被覆寫的類成員,並是一個子類的執行個體,子類的成員將被訪問(而不是訪問父類中的成員).

看圖1. 這段代碼輸出” Hey! I am Son.” 因為當PHP調用getSalutation, 是一個Son的執行個體,是將Father中的salutation覆寫而來. 如果salutation是public的,PHP將產生相同的結果. 覆寫方法的操作很類似.在Son中,對於identify的調用綁定到那個方法.

即使在子類中訪問方式被從protected削弱成public, 動態綁定仍然會發生. 按照訪問方式使用的原則,增強對於類成員的訪問限制是不可能的. 所以把訪問方式從public改變成protected不可能進行.

動態綁定

<?php
class Father
{
protected $salutation = "Hello there!"; //問候

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

protected function identify()
{
print("I am Father.");
}
};

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

protected function identify() //父類中的protected identify() 被覆寫
{
print("I am Son.");
}
};

$obj = new Son();
$obj->getSalutation(); //輸出Hey! I am Son.
?>

//注: 在子類中沒有覆寫getSalutation(),但實際上仍然存在一getSalutation().這個類中的$salutation和identify()

//與Son子類的執行個體中的getSalutation()方法動態綁定,所以調用Son的執行個體的getSalutation()方法,

//將調用Son類中的成員salutation及identify(),而不是父類中的成員salutation及identify().

Private成員只存在於它們所在的類內部. 不像public和protected成員那樣,PHP類比靜態繫結. 看例子圖2 . 它輸出”Hello there! I am Father.”,儘管子類覆寫了salutation的值. 指令碼將this->salutation和當前類Father綁定. 類似的原則應用於private方法identify().

Binding and private members

<?php
class Father
{
private $salutation = "Hello there!";

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

private function identify()
{
print("I am Father.");
}
}

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

$obj = new Son();
$obj->getSalutation(); //輸出Hello there! I am Father.
?>

動態綁定的好處是允許繼承類來改變父類的行為,同時可以保持父類的介面和功能. 看例子圖3. 由於使用了動態綁定,在deleteUser中被調用的isAuthorized的version 可以由對象的類型來確定. 如果是一個普通的user,PHP調用User::isAuthorized會返回FALSE.如果是一個AuthorizedUser的執行個體,PHP調用AuthorizedUser::isAuthorized,將允許deleteUser順利執行.

//haohappy注:用一句話說清楚,就是物件類型與方法,屬性綁定. 調用一個父類與子類中都存在的方法或訪問一個屬性時,會先判斷執行個體屬於哪種物件類型,再調用相應的類中的方法和屬性.

動態綁定的好處

<?php
class User //使用者
{
protected function isAuthorized() //是否是驗證使用者
{
return(FALSE);
}

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

public function deleteUser($username) //刪除使用者
{
if(!$this->isAuthorized())
{
print("You are not authorized.");
return(FALSE);
}

//delete the user
print("User deleted.");
}
}

class AuthorizedUser extends User //認證使用者
{
protected function isAuthorized() //覆寫isAuthorized()
{
return(TRUE);
}
}

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

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

//authorized
$admin->deleteUser("Zeev");
?>

為什麼private的類成員類比靜態繫結? 為了回答這個問題, 你需要回憶一下為什麼需要有private成員.什麼時候用它們來代替protected成員是有意義的?

private成員只有當你不想讓子類繼承改變或特殊化父類的行為時才用到. 這種情況比你想像的要少. 通常來說,一個好的對象分層結構應當允許絕大多數功能被子類特殊化,改進,或改變?這是物件導向編程的基礎之一. 一定的情況下需要private方法或變數,例如當你確信你不想允許子類改變父類中的某個特定的部份.

http://www.bkjia.com/PHPjc/486581.htmlwww.bkjia.comtruehttp://www.bkjia.com/PHPjc/486581.htmlTechArticle除了限制訪問,訪問方式也決定哪個方法將被子類調用或哪個屬性將被子類訪問. 函數調用與函數本身的關聯,以及成員訪問與變數記憶體位址間...

  • 相關文章

    聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

    如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

    A Free Trial That Lets You Build Big!

    Start building with 50+ products and up to 12 months usage for Elastic Compute Service

    • Sales Support

      1 on 1 presale consultation

    • After-Sales Support

      24/7 Technical Support 6 Free Tickets per Quarter Faster Response

    • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.