PHP 5.0物件模型深度探索之訪問方式

來源:互聯網
上載者:User
關鍵字 PHP 5.0物件模型深度探索之訪問方式

php5的訪問方式允許限制對類成員的訪問。這是在PHP5中新增的功能,但在許多物件導向語言中都早已存在。有了訪問方式,才能開發一個可靠的物件導向應用程式,並且構建可重用的物件導向類庫。

像C++和java一樣,PHP有三種訪問方式:public,PRivate和protected. 對於一個類成員的訪問方式,可以是其中之一. 如果你沒有指明訪問方式,預設地訪問方式為public. 你也可以為靜態成員指明一種訪問方式,將訪問方式放在static關鍵字之前(如public static).

Public成員可以被毫無限制地訪問.類外部的任何代碼都可以讀寫public屬性. 你可以從指令碼的任何地方調用一個public方法。在PHP的前幾個版本中,所有方法和屬性都是public, 這讓人覺得對象就像是結構精巧的數組。

Private(私人)成員只在類的內部可見,你不能在一個private屬性所在的類方法之外改變或讀取它的值。同樣地,只有在同一個類中的方法可以調用一個private方法,繼承的子類也不能訪問父類中的private 成員。

要注意,類中的任何成員和類的執行個體都可以訪問private成員。看例子6.8,equals方法將兩個widget進行比較.==運算子比較同一個類的兩個對象,但這個例子中每個對象執行個體都有唯一的ID.equals方法只比較name和price。注意equals方法如何訪問另一個Widget執行個體的private屬性,Java和C都允許這樣的操作。

Listing 6.8 Private members

class Widget
{
 private $name;
 private $price;
 private $id;

 public function __construct($name, $price)
 {
  $this->name = $name;
  $this->price = floatval($price);
  $this->id = uniqid();
 }
 //checks if two widgets are the same 檢查兩個widget是否相同
 public function equals($widget)
 {
  return(($this->name == $widget->name)AND ($this->price == $widget->price));
 }
}
$w1 = new Widget('Cog', 5.00);
$w2 = new Widget('Cog', 5.00);
$w3 = new Widget('Gear', 7.00);

//TRUE
if($w1->equals($w2))
{
 print("w1 and w2 are the same n");
}

//FALSE
if($w1->equals($w3))
{
print("w1 and w3 are the same n");
}

//FALSE, == includes id in comparison
if($w1 == $w2) //不等,因為ID不同
{
print("w1 and w2 are the same n");
}
?>

如果你對物件導向編程不熟悉,你可能想知道用private成員的目的是什麼. 你可以回憶一下封裝和耦合的想法,這在本章開頭我們有討論過。Private成員有助於封裝資料,他們可以隱藏在一個類內部而不被類外部的代碼接觸到,同時他們還有助於實現鬆散的耦合. 如果資料結構外的代碼不能直接存取內部屬性,那麼就不會產生一個隱性的關聯性。

當然,大部分private屬性仍然可以被外部代碼共用. 解決方案是用一對public方法,一個是get(擷取屬性的值),另一個是set(設定屬性的值)。建構函式也接受屬性的初始值. 這使得成員間的交流通過一個狹窄的,經過良好限定的介面來進行. 這也提供改變傳遞給方法的值的機會,注意在例子6.8中,建構函式如何強制使price成為一個float數(floadval())。

Protected(受保護的) 成員能被同個類中的所有方法和繼承出的類的中所有方法訪問到。Public屬性有違封裝的精神,因為它們允許子類依賴於一個特定的屬性來書寫,protected方法則不會帶來這方面的擔憂,一個使用protected方法的子類需要很清楚它的父類的結構才行。

例子6.9由例子6.8改進而得到,包含了一個Widget的子類Thing。注意Widget現在有一個叫作getName的protected方法. 如果Widget的執行個體試圖調用protected方法將會出錯: $w1->getName()產生了一個錯誤, 但子類Thing中的getName方法可以調用這個protected方法.當然對於證明Widget::getName方法是protected,這個例子顯得過於簡單,在實際情況下,使用protected方法要依賴於對對象的內部結構的理解。

Listing 6.9 Protected members

class Widget
{
 private $name;
 private $price;
 private $id;

 public function __construct($name, $price)
 {
  $this->name = $name;
  $this->price = floatval($price);
  $this->id = uniqid();
 }

 //checks if two widgets are the same
 public function equals($widget)
 {
  return(($this->name == $widget->name)AND($this->price == $widget->price));
 }

 protected function getName()
 {
  return($this->name);
 }
}

class Thing extends Widget
{
 private $color;

 public function setColor($color)
 {
  $this->color = $color;
 }

 public function getColor()
 {
  return($this->color);
 }

 public function getName()
 {
  return(parent::getName());
 }
}

$w1 = new Widget('Cog', 5.00);
$w2 = new Thing('Cog', 5.00);
$w2->setColor('Yellow');

//TRUE (still!) 結果仍然為真
if($w1->equals($w2))
{
 print("w1 and w2 are the same n");
}

//print Cog 輸出 Cog
print($w2->getName());
?>

一個子類可能改變通過覆寫父類方法來改變方法的訪問方式,儘管如此,仍然有一些限制。如果你覆寫了一個public類成員,他子類中必須保持public. 如果你覆寫了一個protected成員,它可保持protected或變成public.Private成員仍然只在當前類中可見。聲明一個與父類的private成員同名的成員將簡單地在當前類中建立一個與原來不同的成員. 因此,在技術上你不能覆寫一個private成員。

Final關鍵字是限制訪問成員方法的另一個方法。子類不能覆寫父類中標識為final的方法,Final關鍵字不能用於屬性。

  • 相關文章

    聯繫我們

    該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在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.