新聞中心
PHP 5中引入了對(duì)象重載技術(shù),本文將探討對(duì)于方法__call(),__set()以及__get()進(jìn)行重載的可能性。在對(duì)重載理論作簡(jiǎn)單介紹后,我們將通過(guò)兩個(gè)例子直奔主題:***例,實(shí)現(xiàn)持續(xù)存儲(chǔ)類;第二例,找到一種實(shí)現(xiàn)動(dòng)態(tài)的getter/setter的方法。

我們提供的服務(wù)有:成都網(wǎng)站設(shè)計(jì)、網(wǎng)站制作、微信公眾號(hào)開(kāi)發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、來(lái)安ssl等。為上1000+企事業(yè)單位解決了網(wǎng)站和推廣的問(wèn)題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的來(lái)安網(wǎng)站制作公司
關(guān)于PHP更多內(nèi)容,推薦專題:PHP開(kāi)發(fā)基礎(chǔ)入門(mén)
一、什么是對(duì)象重載?
在PHP中談到對(duì)象重載時(shí),我們要區(qū)別兩種類型:
◆方法重載
◆屬性重載
在方法重載的情況下,我們要定義一個(gè)魔術(shù)般的方法__call(),它將實(shí)現(xiàn)一個(gè)在相應(yīng)類中對(duì)未定義方法的籠統(tǒng)調(diào)用。只有當(dāng)你想存取類中未定義的方法時(shí),這種籠統(tǒng)方法才會(huì)被調(diào)用。在沒(méi)有方法重載的情況下,下面的例子將導(dǎo)致PHP顯示一條致命錯(cuò)誤信息:Call to undefined method ThisWillFail::bar() in/some/directory/example.php on line 9并流產(chǎn)程序的執(zhí)行:
- <?php
- class ThisWillFail {
- public function foo() {
- return "Hello World!";
- }
- }
- $class = new ThisWillFail;
- $class->bar();
- ?>
借助方法重載的幫助,代碼能夠捕獲到這種調(diào)用且能夠體面地給以處理。屬性重載與方法重載差不多。這種情況下,類把讀/寫(xiě)操作重定向(亦可稱代理)到類的屬性,這些屬性在類中沒(méi)有顯式定義。這里的專門(mén)方法是__set()和__get()。依賴于錯(cuò)誤報(bào)告等級(jí),PHP翻譯器通常在存取一個(gè)未定義的屬性時(shí),或者發(fā)出一個(gè)通知,或者推遲一下并潛在地定義這個(gè)變量。
而如果使用屬性重載,翻譯器卻可以在設(shè)置一個(gè)未定義的屬性時(shí)調(diào)用__set(),而在存取一個(gè)未定義的屬性值時(shí)調(diào)用__get()。綜上所述,利用重載技術(shù)可以實(shí)現(xiàn)在象用PHP這樣的動(dòng)態(tài)語(yǔ)言進(jìn)行時(shí)軟件開(kāi)發(fā)時(shí)間的大大縮短。
二、持續(xù)性存儲(chǔ)類舉例
下列代碼,通過(guò)使用屬性重載技術(shù),用少于50行的PHP代碼實(shí)現(xiàn)了上面所提到的持續(xù)性存儲(chǔ)類。術(shù)語(yǔ)persistable意味著類可以從一個(gè)數(shù)據(jù)結(jié)構(gòu)中描述一個(gè)元素,并保持與底端存儲(chǔ)系統(tǒng)的同步。用編碼的解釋就是,外部代碼可以使用類來(lái)實(shí)現(xiàn)從一個(gè)數(shù)據(jù)庫(kù)表中選定一行。
這樣,在程序運(yùn)行時(shí),可以直接存取類的屬性來(lái)操縱該行中的元素(讀/取)。在腳本結(jié)束時(shí),PHP將負(fù)責(zé)把更新的行數(shù)據(jù)回送到數(shù)據(jù)庫(kù)中去。精心研讀下面代碼將有助于你理解什么是屬性重載。
- <?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 = "john@example.com";
- ?>
#p#
你遇到的***個(gè)問(wèn)題可能是__construct(),這是PHP 5中引入的新的構(gòu)造器方法。在PHP 4時(shí)代,構(gòu)造器總是與它們的類名相匹配。在PHP 5中已不再是這樣。你不需要對(duì)構(gòu)造器方法有過(guò)多的了解,除了調(diào)用它可以創(chuàng)建一個(gè)類的實(shí)例外;并注意到,這里使用了一個(gè)參數(shù) - 執(zhí)行一個(gè)基于此參數(shù)的數(shù)據(jù)庫(kù)。此構(gòu)造器把查詢結(jié)果賦值給類屬性$data。
接下來(lái),程序定義了兩個(gè)特別的方法__get()和__set()。你應(yīng)該對(duì)它們?cè)缫咽煜ぃ篲_get()用于讀取未定義的屬性值,__set()用于修改未定義的屬性值。
這意味著無(wú)論什么時(shí)候從持續(xù)性存儲(chǔ)類中讀取/寫(xiě)入一個(gè)未定義的屬性,由這些專門(mén)方法來(lái)負(fù)責(zé)管理在屬性數(shù)組變量$data中的信息,而不是直接改變類的屬性(切記:變量$data包含著來(lái)自于數(shù)據(jù)庫(kù)中的一行!)。
類中的***一個(gè)方法是__construct()的對(duì)立者- 析構(gòu)器__destruct()。PHP在"腳本關(guān)閉階段"調(diào)用析構(gòu)器,典型地這是在PHP腳本執(zhí)行快要結(jié)束的時(shí)候。析構(gòu)器把來(lái)自于$data屬性的信息寫(xiě)回到數(shù)據(jù)庫(kù)中去。這正是前面同步(synchronization )術(shù)語(yǔ)的含義。
你可能早已注意到,這里的代碼使用了PEAR的數(shù)據(jù)庫(kù)抽象層包(database abstraction layer package)。其實(shí)這無(wú)所謂,通過(guò)別的方式與數(shù)據(jù)庫(kù)通訊也一樣能說(shuō)明本文的主題。
如果你細(xì)心觀察,會(huì)發(fā)現(xiàn)該持續(xù)性存儲(chǔ)類的描述比較簡(jiǎn)單。例子中僅涉及了一個(gè)數(shù)據(jù)庫(kù)表,而沒(méi)有考慮更復(fù)雜的數(shù)據(jù)模型,如使用LEFT JOIN和其它復(fù)雜的數(shù)據(jù)庫(kù)操作技術(shù)。然而你不必受此約束,借助于屬性重載,你可以使用你自己理想的數(shù)據(jù)庫(kù)模型。只需要加入少許代碼,你即可以在該持續(xù)性存儲(chǔ)類中運(yùn)用復(fù)雜的數(shù)據(jù)庫(kù)特性。
還存在一個(gè)小問(wèn)題 - 當(dāng)在析構(gòu)器中查詢失敗時(shí)并沒(méi)有引入錯(cuò)誤處理機(jī)制。是析構(gòu)器的天性導(dǎo)致在這種情況下不可能顯示相應(yīng)的錯(cuò)誤信息,因?yàn)闃?gòu)建HTML標(biāo)志常常在PHP調(diào)用構(gòu)析器之前就已經(jīng)結(jié)束了。
為解決這個(gè)問(wèn)題,你可以把__destruct()重命名為象saveData()這樣的名字并在調(diào)用腳本的某處手工執(zhí)行這一方法。這對(duì)于類的持續(xù)性存儲(chǔ)的概念并沒(méi)有任何改變;僅是多寫(xiě)幾行代碼而已。作為選擇,你還可以在析構(gòu)器中使用函數(shù)error_log()來(lái)記錄下屬于系統(tǒng)范圍的錯(cuò)誤記錄文件中的錯(cuò)誤信息。屬性重載的工作機(jī)制就是這樣。下面我們討論一下方法重載。
三、方法重載舉例
1. 動(dòng)態(tài)的Getter/Setter方法
下列代碼實(shí)現(xiàn)了"動(dòng)態(tài)"getter/setter方法以借助于方法重載的幫助來(lái)控制類。下面我們結(jié)合源代碼進(jìn)行分析:
- <?php
- class DynamicGetterSetter {
- private $name = "Martin Jansen";
- private $starbucksdrink = "Caramel Cappuccino Swirl";
- func
- tion __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() . "\n\n";
- $class->setName("John Doe");
- $class->setStarbucksDrink("Classic Coffee");
- echo "Name: " . $class->getName() . "\n";
- echo "Favourite Starbucks flavour: " . $class->getStarbucksDrink() . "\n\n";
- ?>
很明顯,這里的兩個(gè)屬性$name和$starbucksdrink都是私有的,就是說(shuō)從類的外部是不能夠存取這些屬性的。在面向?qū)ο蟮木幊讨校瑢?shí)現(xiàn)公共的getter/setter方法來(lái)存取或修改非公共屬性的值是很經(jīng)常的事情。實(shí)現(xiàn)這些是單調(diào)的事情,且相當(dāng)耗費(fèi)時(shí)間和精力。
借助于方法重載可以容易得解決這個(gè)問(wèn)題。不是為每個(gè)屬性實(shí)現(xiàn)getter/setter方法,上面只實(shí)現(xiàn)了一個(gè)通用的__call()方法。這意味著當(dāng)調(diào)用一個(gè)未定義的getter/setter方法如setName()或者getStarbucksdrink()時(shí),PHP不會(huì)產(chǎn)生一個(gè)致命錯(cuò)誤而流產(chǎn),而是執(zhí)行(或者代理到)魔術(shù)般的__call()方法。這是些簡(jiǎn)單介紹,下面我們對(duì)__call()作一下深入分析。
2. 詳細(xì)分析__call()方法
__call()的***個(gè)參數(shù)是原始的且尚未確定的方法(如setName),第二個(gè)參數(shù)是一個(gè)數(shù)字索引的一維數(shù)組,它包含了原始方法的所有參數(shù)。用兩個(gè)參數(shù)("Martin"和42)調(diào)用一個(gè)未定義的方法將產(chǎn)生下面數(shù)組:
- $class->thisMethodDoesNotExist("Martin", 42);
- /導(dǎo)向__call()的第二個(gè)參數(shù)
- Array
- (
- [0] => Martin
- [1] => 42
- )
在方法__call()內(nèi)部,如果原始方法以get或者set開(kāi)頭,則要進(jìn)行某種計(jì)算以確定是否代碼調(diào)用的是一個(gè)getter/setter方法。而且,這種方法還要進(jìn)一步分析方法名的另外一組成部分(除去開(kāi)始的三個(gè)字符),因?yàn)楹竺孢@部分字符串正代表getter/setter參照的屬性的名字。
如果方法名中指示有一個(gè)getter/setter,那么該方法或者返回相應(yīng)的屬性值,或者設(shè)置原始方法的***個(gè)參數(shù)的值。如果沒(méi)有的話,它不做任何事情,繼續(xù)執(zhí)行程序,好象沒(méi)有事情發(fā)生。
3. 實(shí)現(xiàn)目標(biāo)
實(shí)質(zhì)上,相應(yīng)于任意的屬性,存在一種方法允許代碼動(dòng)態(tài)地調(diào)用任意的getter/setter方法,這種算法是存在的。這在短期內(nèi)開(kāi)發(fā)一個(gè)程序原型的情況下是很方便的:不是花費(fèi)大量時(shí)間來(lái)實(shí)現(xiàn)getters/setters,開(kāi)發(fā)人員可以專注于建模API并保證應(yīng)用程序的根本正確。把__call()方法納入到一個(gè)抽象類中甚至有可能使你在將來(lái)的PHP工程開(kāi)發(fā)中實(shí)現(xiàn)代碼的重用!
4. 不足之外
有優(yōu)點(diǎn)就有缺點(diǎn)。以上方法也有幾個(gè)不足:較大些的項(xiàng)目可以會(huì)使用象phpDocumentor這樣的工具來(lái)跟蹤API結(jié)構(gòu)。用上面介紹的動(dòng)態(tài)方法,所有的getter/setter方法當(dāng)然不會(huì)出現(xiàn)在自動(dòng)生成的文檔中,這是無(wú)需多作解釋的。
另外一個(gè)不足是,類外面的代碼可以存取類內(nèi)的每一個(gè)私有屬性。當(dāng)使用真正的getter/setter方法時(shí),有可能區(qū)別開(kāi)外部代碼可以存取的私有屬性和對(duì)類外部不可見(jiàn)的"真正的"私有屬性 - 因?yàn)槲覀冇蟹椒ㄖ剌d,而且有虛擬的getter和setter方法可以利用。
結(jié)論
本文通過(guò)兩個(gè)例子細(xì)致分析了PHP 5中對(duì)象重載的兩種情形。很希望本文的方法幫助你提高PHP編程的工作效率!同時(shí),你也應(yīng)清醒地看到這種方法的不足。
【編輯推薦】
- 重載和類的自動(dòng)加載
- PHP面向?qū)ο缶幊痰幕A(chǔ)知識(shí)講解
- PHP 5 數(shù)據(jù)對(duì)象 (PDO) 抽象層與 Oracle
網(wǎng)站名稱:探秘PHP 5的對(duì)象重載技術(shù)
轉(zhuǎn)載來(lái)于:http://www.dlmjj.cn/article/djeojdc.html


咨詢
建站咨詢
