探秘PHP 5的對象重載技術

來源:互聯網
上載者:User

PHP 5中引入了對象重載技術,本文將探討對於方法__call(),__set()以及__get()進行重載的可能性。在對重載理論作簡單介紹後,我們將通過兩個例子直奔主題:第一例,實現持續儲存類;第二例,找到一種實現動態getter/setter的方法。

 

一、什麼是對象重載?

在PHP中談到對象重載時,我們要區別兩種類型:

◆方法重載

◆屬性重載

在方法重載的情況下,我們要定義一個魔術般的方法__call(),它將實現一個在相應類中對未定義方法的籠統調用。只有當你想存取類中未定義的方法時,這種籠統方法才會被調用。在沒有方法重載的情況下,下面的例子將導致PHP顯示一條致命錯誤資訊:Call to undefined method ThisWillFail::bar() in/some/directory/example.php on line 9併流產程式的執行:


  1. <?php  
  2.  class ThisWillFail {  
  3. public function foo() {  
  4.  return "Hello World!";  
  5. }  
  6.  }  
  7.  $class = new ThisWillFail;  
  8.  $class->bar();  
  9. ?>  

藉助方法重載的協助,代碼能夠捕獲到這種調用且能夠體面地給以處理。屬性重載與方法重載差不多。這種情況下,類把讀/寫操作重新導向(亦可稱代理)到類的屬性,這些屬性在類中沒有顯式定義。這裡的專門方法是__set()和__get()。依賴於錯誤報表等級,PHP翻譯器通常在存取一個未定義的屬性時,或者發出一個通知,或者延遲一下並潛在地定義這個變數。

而如果使用屬性重載,翻譯器卻可以在設定一個未定義的屬性時調用__set(),而在存取一個未定義的屬性值時調用__get()。綜上所述,利用重載技術可以實現在象用PHP這樣的動態語言進行時軟體開發時間的大大縮短。

二、持久性儲存類舉例

下列代碼,通過使用屬性重載技術,用少於50行的PHP代碼實現了上面所提到的持久性儲存類。術語persistable意味著類可以從一個資料結構中描述一個元素,並保持與底端儲存系統的同步。用編碼的解釋就是,外部代碼可以使用類來實現從一個資料庫表中選定一行。

這樣,在程式運行時,可以直接存取類的屬性來操縱該行中的元素(讀/取)。在指令碼結束時,PHP將負責把更新的行資料回送到資料庫中去。精心研讀下面代碼將有助於你理解什麼是屬性重載。


  1. <?php  
  2.  //裝入PEAR的 <a href="http://pear.php.net/package/DB/">DB package</a>  
  3.  require_once "DB.php";  
  4.  class Persistable {  
  5. private $data = array();  
  6. private $table = "users";  
  7. public function __construct($user) {  
  8.  $this->dbh = DB::Connect("mysql://user:password@localhost/database");  
  9.  $query = "SELECT id, name, email, country FROM " .  
  10.  $this->table . " WHERE name = ?";  
  11.  $this->data = $this->dbh->getRow($query, array($user),  
  12.  DB_FETCHMODE_ASSOC);  
  13. }  
  14. public function __get($member) {  
  15.  if (isset($this->data[$member])) {  
  16. return $this->data[$member];  
  17.  }  
  18. }  
  19.  
  20. public function __set($member, $value) {  
  21.  // dataset的ID是唯讀  
  22.  if ($member == "id") {  
  23. return;  
  24.  }  
  25.  if (isset($this->data[$member])) {  
  26. $this->data[$member] = $value;  
  27.  }  
  28. }  
  29. public function __destruct() {  
  30.  $query = "UPDATE " . $this->table . " SET name = ?,   
  31.  email = ?, country = ? WHERE id = ?";  
  32.  $this->dbh->query($query, $this->name, $this->email,   
  33.  $this->country, $this->id);  
  34. }  
  35.  }  
  36.  $class = new Persistable("Martin Jansen");  
  37.  $class->name = "John Doe";  
  38.  $class->country = "United States";  
  39.  $class->email = "john@example.com";  
  40. ?>  

你遇到的第一個問題可能是__construct(),這是PHP 5中引入的新的構造器方法。在PHP 4時代,構造器總是與它們的類名相匹配。在PHP 5中已不再是這樣。你不需要對構造器方法有過多的瞭解,除了調用它可以建立一個類的執行個體外;並注意到,這裡使用了一個參數 - 執行一個基於此參數的資料庫。此構造器把查詢結果賦值給類屬性$data。

接下來,程式定義了兩個特別的方法__get()和__set()。你應該對它們早已熟悉:__get()用於讀取未定義的屬性值,__set()用於修改未定義的屬性值。

這意味著無論什麼時候從持久性儲存類中讀取/寫入一個未定義的屬性,由這些專門方法來負責管理在屬性陣列變數$data中的資訊,而不是直接改變類的屬性(切記:變數$data包含著來自於資料庫中的一行!)。

類中的最後一個方法是__construct()的對立者- 析構器__destruct()。PHP在"指令碼關閉階段"調用析構器,典型地這是在PHP指令碼執行快要結束的時候。析構器把來自於$data屬性的資訊寫回到資料庫中去。這正是前面同步(synchronization )術語的含義。

你可能早已注意到,這裡的代碼使用了PEAR的資料庫抽象層包(database abstraction layer package)。其實這無所謂,通過別的方式與資料庫通訊也一樣能說明本文的主題。

如果你細心觀察,會發現該持久性儲存類的描述比較簡單。例子中僅涉及了一個資料庫表,而沒有考慮更複雜的資料模型,如使用LEFT JOIN和其它複雜的資料庫操作技術。然而你不必受此約束,藉助於屬性重載,你可以使用你自己理想的資料庫模型。只需要加入少許代碼,你即可以在該持久性儲存類中運用複雜的資料庫特性。

還存在一個小問題 - 當在析構器中查詢失敗時並沒有引入錯誤處理機制。是析構器的天性導致在這種情況下不可能顯示相應的錯誤資訊,因為構建HTML標誌常常在PHP調用構析器之前就已經結束了。

為解決這個問題,你可以把__destruct()重新命名為象saveData()這樣的名字並在呼叫指令碼的某處手工執行這一方法。這對於類的持久性儲存的概念並沒有任何改變;僅是多寫幾行代碼而已。作為選擇,你還可以在析構器中使用函數error_log()來記錄下屬於系統範圍的錯誤記錄檔案中的錯誤資訊。屬性重載的工作機制就是這樣。下面我們討論一下方法重載。

 三、方法重載舉例

1. 動態Getter/Setter方法

下列代碼實現了"動態"getter/setter方法以藉助於方法重載的協助來控制類。下面我們結合原始碼進行分析:


  1. <?php  
  2.  class DynamicGetterSetter {  
  3. private $name = "Martin Jansen";  
  4. private $starbucksdrink = "Caramel Cappuccino Swirl";  
  5. func  

  1. tion __call($method, $arguments) {  
  2.  $prefix = strtolower(substr($method, 0, 3));  
  3.  $property = strtolower(substr($method, 3));  
  4.  if (empty($prefix)  empty($property)) {  
  5. return;  
  6.  }  
  7.  if ($prefix == "get" && isset($this->$property)) {  
  8. return $this->$property;  
  9.  }  
  10.  if ($prefix == "set") {  
  11. $this->$property = $arguments[0];  
  12.  }  
  13. }  
  14.  }  
  15.  $class = new DynamicGetterSetter;  
  16.  echo "Name: " . $class->getName() . "\n";  
  17.  echo "Favourite Starbucks flavour: " . $class->getStarbucksDrink() . "\n\n";  
  18.  $class->setName("John Doe");  
  19.  $class->setStarbucksDrink("Classic Coffee");  
  20.  echo "Name: " . $class->getName() . "\n";  
  21.  echo "Favourite Starbucks flavour: " . $class->getStarbucksDrink() . "\n\n";  
  22. ?>  

很明顯,這裡的兩個屬性$name和$starbucksdrink都是私人的,就是說從類的外部是不能夠存取這些屬性的。在物件導向的編程中,實現公用的getter/setter方法來存取或修改非公用屬性的值是很經常的事情。實現這些是單調的事情,且相當耗費時間和精力。

藉助於方法重載可以容易得解決這個問題。不是為每個屬性實現getter/setter方法,上面只實現了一個通用的__call()方法。這意味著當調用一個未定義的getter/setter方法如setName()或者getStarbucksdrink()時,PHP不會產生一個致命錯誤而流產,而是執行(或者代理到)魔術般的__call()方法。這是些簡單介紹,下面我們對__call()作一下深入分析。

2. 詳細分析__call()方法

__call()的第一個參數是原始的且尚未確定的方法(如setName),第二個參數是一個數字索引的一維數組,它包含了原始方法的所有參數。用兩個參數("Martin"和42)調用一個未定義的方法將產生下面數組:


  1. $class->thisMethodDoesNotExist("Martin", 42);  
  2. /導向__call()的第二個參數  
  3. Array  
  4. (  
  5. [0] => Martin  
  6. [1] => 42  
  7. )  

在方法__call()內部,如果原始方法以get或者set開頭,則要進行某種計算以確定是否代碼調用的是一個getter/setter方法。而且,這種方法還要進一步分析方法名的另外一組成部分(除去開始的三個字元),因為後面這部分字串正代表getter/setter參照的屬性的名字。

如果方法名中指示有一個getter/setter,那麼該方法或者返回相應的屬性值,或者設定原始方法的第一個參數的值。如果沒有的話,它不做任何事情,繼續執行程式,好象沒有事情發生。

3. 實現目標

實質上,相應於任意的屬性,存在一種方法允許代碼動態地調用任意的getter/setter方法,這種演算法是存在的。這在短期內開發一個程式原型的情況下是很方便的:不是花費大量時間來實現getters/setters,開發人員可以專註於建模API並保證應用程式的根本正確。把__call()方法納入到一個抽象類別中甚至有可能使你在將來的PHP工程開發中實現代碼的重用!

4. 不足之外

有優點就有缺點。以上方法也有幾個不足:較大些的項目可以會使用象phpDocumentor這樣的工具來跟蹤API結構。用上面介紹的動態方法,所有的getter/setter方法當然不會出現在自動產生的文檔中,這是無需多作解釋的。

另外一個不足是,類外面的代碼可以存取類內的每一個私人屬性。當使用真正的getter/setter方法時,有可能區別開外部代碼可以存取的私人屬性和對類外部不可見的"真正的"私人屬性 - 因為我們有方法重載,而且有虛擬getter和setter方法可以利用。

結論

本文通過兩個例子細緻分析了PHP 5中對象重載的兩種情形。很希望本文的方法協助你提高PHP編程的工作效率!同時,你也應清醒地看到這種方法的不足。



相關文章

Beyond APAC's No.1 Cloud

19.6% IaaS Market Share in Asia Pacific - Gartner IT Service report, 2018

Learn more >

Apsara Conference 2019

The Rise of Data Intelligence, September 25th - 27th, Hangzhou, China

Learn more >

Alibaba Cloud Free Trial

Learn and experience the power of Alibaba Cloud with a free trial worth $300-1200 USD

Learn more >

聯繫我們

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

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