久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術文章
文章詳情頁

PHP 5.0 中的對象重載技術研究

瀏覽:6日期:2024-02-02 11:11:09

文/朱先忠編譯

一、簡介

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

二、什么是對象重載?

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

·方法重載

·屬性重載

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

<?php class ThisWillFail {public function foo() { return 'Hello World!';} } $class = new ThisWillFail; $class->bar();?>

借助方法重載的幫助,代碼能夠捕獲到這種調用且能夠體面地給以處理。

屬性重載與方法重載差不多。這種情況下,類把讀/寫操作重定向(亦可稱代理)到類的屬性,這些屬性在類中沒有顯式定義。這里的專門方法是__set()和__get()。依賴于錯誤報告等級,PHP翻譯器通常在存取一個未定義的屬性時,或者發出一個通知,或者推遲一下并潛在地定義這個變量。而如果使用屬性重載,翻譯器卻可以在設置一個未定義的屬性時調用__set(),而在存取一個未定義的屬性值時調用__get()。綜上所述,利用重載技術可以實現在象用PHP這樣的動態語言進行時軟件開發時間的大大縮短。

理論介紹至此,下面分析具體編碼。

三、持續性存儲類舉例

下列代碼,通過使用屬性重載技術,用少于50行的PHP代碼實現了上面所提到的持續性存儲類。術語persistable意味著類可以從一個數據結構中描述一個元素,并保持與底端存儲系統的同步。用編碼的解釋就是,外部代碼可以使用類來實現從一個數據庫表中選定一行。這樣,在程序運行時,可以直接存取類的屬性來操縱該行中的元素(讀/取)。在腳本結束時,PHP將負責把更新的行數據回送到數據庫中去。

精心研讀下面代碼將有助于你理解什么是屬性重載。

<?php //裝入PEAR的 <a >DB package</a> require_once 'DB.php'; class Persistable {private $data = array();private $table = 'users';public function __construct($user) { $this->dbh = DB::Connect('mysql://user:password@localhost/database'); $query = 'SELECT id, name, email, country FROM ' . $this->table . ' WHERE name = ?'; $this->data = $this->dbh->getRow($query, array($user), DB_FETCHMODE_ASSOC);}public function __get($member) { if (isset($this->data[$member])) {return $this->data[$member]; }}public function __set($member, $value) { // dataset的ID是只讀的 if ($member == 'id') {return; } if (isset($this->data[$member])) {$this->data[$member] = $value; }}public function __destruct() { $query = 'UPDATE ' . $this->table . ' SET name = ?,  email = ?, country = ? WHERE id = ?'; $this->dbh->query($query, $this->name, $this->email,  $this->country, $this->id);} } $class = new Persistable('Martin Jansen'); $class->name = 'John Doe'; $class->country = 'United States'; $class->email = '[email protected]';?>

你遇到的第一個問題可能是__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方法以借助于方法重載的幫助來控制類。下面我們結合源代碼進行分析:

<?php class DynamicGetterSetter {private $name = 'Martin Jansen';private $starbucksdrink = 'Caramel Cappuccino Swirl';function __call($method, $arguments) { $prefix = strtolower(substr($method, 0, 3)); $property = strtolower(substr($method, 3)); if (empty($prefix) || empty($property)) {return; } if ($prefix == 'get' && isset($this->$property)) {return $this->$property; } if ($prefix == 'set') {$this->$property = $arguments[0]; }} } $class = new DynamicGetterSetter; echo 'Name: ' . $class->getName() . 'n'; echo 'Favourite Starbucks flavour: ' . $class->getStarbucksDrink() . 'nn'; $class->setName('John Doe'); $class->setStarbucksDrink('Classic Coffee'); echo 'Name: ' . $class->getName() . 'n'; echo 'Favourite Starbucks flavour: ' . $class->getStarbucksDrink() . 'nn';?>

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

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

這是些簡單介紹,下面我們對__call()作一下深入分析。

2. 詳細分析__call()方法

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

$class->thisMethodDoesNotExist('Martin', 42);/導向__call()的第二個參數Array([0] => Martin[1] => 42)

在方法__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.0中對象重載的兩種情形。很希望本文的方法幫助你提高PHP編程的工作效率!同時,你也應清醒地看到這種方法的不足。

標簽: PHP
主站蜘蛛池模板: 成人免费视频观看视频 | www.44181com| 国产精品久久久久久久竹霞 | 国产成人久久精品77777 | 国产精品久久久久国产精品 | 视频在线一区二区三区 | 久久国产精品久久久久久电车 | 91精品一区二区三区久久久久久 | 欧美日韩国产一区二区三区不卡 | 中文字幕在线观看 | 国产一二三区在线播放 | 亚洲三级在线看 | 中文字幕2021| 看片wwwwwwwwwww| 精品 99| 亚洲免费观看视频 | 国产精品不卡 | 欧美久久久久久 | 国产日产精品一区二区三区四区 | 中国大陆高清aⅴ毛片 | 国产亚洲精品美女久久久久久久久久 | 日韩视频三区 | 国产一区二区三区高清 | 99精品欧美一区二区三区综合在线 | 国产高清一区二区三区 | 欧美 日韩 国产 一区 | 亚洲精品一区在线观看 | 国产精品第一区 | xxxx免费视频| 国产精品美女久久久久久久久久久 | av免费观看网站 | 免费视频久久 | 色视频在线播放 | 欧美久久精品 | 久久精品国产v日韩v亚洲 | 国产视频精品一区二区三区 | 色约约精品免费看视频 | 亚洲欧美视频 | aaa久久| 操操网 | 欧美日韩成人在线视频 |