PHP 5.0對(duì)象模型深度探索之起步
面向?qū)ο缶幊瘫辉O(shè)計(jì)來(lái)為大型軟件項(xiàng)目提供解決方案,尤其是多人合作的項(xiàng)目. 當(dāng)源代碼增長(zhǎng)到一萬(wàn)行甚至更多的時(shí)候,每一個(gè)更動(dòng)都可能導(dǎo)致不希望的副作用. 這種情況發(fā)生于模塊間結(jié)成秘密聯(lián)盟的時(shí)候,就像第一次世界大戰(zhàn)前的歐洲。
//haohappy注:喻指模塊間的關(guān)聯(lián)度過(guò)高,相互依賴性太強(qiáng).更動(dòng)一個(gè)模塊導(dǎo)致其它模塊也必須跟著更動(dòng)。
想像一下,如果有一個(gè)用來(lái)處理登錄的模塊允許一個(gè)信用卡處理模塊來(lái)分享它的數(shù)據(jù)庫(kù)連接. 當(dāng)然出發(fā)點(diǎn)是好的,節(jié)省了進(jìn)行另一個(gè)數(shù)據(jù)庫(kù)連接的支出.然而有時(shí),登錄處理模塊改變了其中一個(gè)變量的名字,就可能割斷了兩者間的協(xié)議.導(dǎo)致信用卡模塊的處理出錯(cuò),進(jìn)而導(dǎo)致處理發(fā)票的模塊出錯(cuò). 很快地,體系中所有無(wú)關(guān)的模塊都可能由此出錯(cuò).
因此,我覺(jué)得有點(diǎn)戲劇性地,絕大多數(shù)程序員都對(duì)耦合和封裝心存感激. 耦合是兩個(gè)模塊間依賴程度的量度. 耦合越少越好.我們希望能夠從已有的項(xiàng)目中抽走一個(gè)模塊并在另一個(gè)新項(xiàng)目中使用.
我們也希望在某個(gè)模塊內(nèi)部大規(guī)模的更動(dòng)而不用擔(dān)心對(duì)其他模塊的影響. 封裝的原則可以提供這個(gè)解決方案.模塊被看待成相對(duì)獨(dú)立,并且模塊間的數(shù)據(jù)通信通過(guò)接口來(lái)進(jìn)行. 模塊不通過(guò)彼此的變量名來(lái)窺探另一個(gè)模塊,它們通過(guò)函數(shù)來(lái)禮貌地發(fā)送請(qǐng)求.
封裝是你可以在任何編程語(yǔ)言中使用的一個(gè)原則. 在PHP和許多面向過(guò)程的語(yǔ)言中,可以偷懶是很有誘惑的.沒(méi)有什么可以阻止你通過(guò)模塊來(lái)構(gòu)建一個(gè)假想的WEB. 面向?qū)ο缶幊淌鞘钩绦騿T不會(huì)違背封裝原則的一種方法.
在面向?qū)ο缶幊讨?模塊被組織成一個(gè)個(gè)對(duì)象. 這些對(duì)象擁有方法和屬性. 從抽象的角度來(lái)看,方法是一個(gè)對(duì)象的所做的動(dòng)作,而屬性是對(duì)象的特性.從編程角度來(lái)看,方法就是函數(shù)而屬性是變量. 在一個(gè)理想化的面向?qū)ο篌w系中,每個(gè)部份都是一個(gè)對(duì)象. 體系由對(duì)象及對(duì)象間通過(guò)方法來(lái)形成的聯(lián)系構(gòu)成.
一個(gè)類定義了對(duì)象的屬性. 如果你在烘烤一組甜餅對(duì)象,那么類將會(huì)是甜餅機(jī). 類的屬性和方法是被調(diào)用的成員. 人們可以通過(guò)說(shuō)出數(shù)據(jù)成員或者方法成員來(lái)表達(dá).
每種語(yǔ)言提供了不同的途徑來(lái)訪問(wèn)對(duì)象. PHP從C++中借用概念,提供一個(gè)數(shù)據(jù)類型用來(lái)在一個(gè)標(biāo)識(shí)符下包含函數(shù)和變量。最初設(shè)計(jì)PHP的時(shí)候,甚至PHP3被開(kāi)發(fā)出時(shí),PHP并不打算提供開(kāi)發(fā)超過(guò)10萬(wàn)行代碼的大型項(xiàng)目的能力。隨著PHP和Zend引擎的發(fā)展,開(kāi)發(fā)大型項(xiàng)目變得有可能,但無(wú)論你的項(xiàng)目規(guī)模多大,用類來(lái)書(shū)寫(xiě)你的腳本將可以讓代碼實(shí)現(xiàn)重用。這是一個(gè)好主意,特別當(dāng)你愿意與別人分享你的代碼的時(shí)候。
有關(guān)對(duì)象的想法是計(jì)算機(jī)科學(xué)上最令人興奮的概念之一。開(kāi)始很難掌握它,但我可以保證,一旦你掌握了它,用它的思維來(lái)思考將會(huì)非常自然。
PHP5 的對(duì)象模型
PHP5有一個(gè)單重繼承的,限制訪問(wèn)的,可以重載的對(duì)象模型. 本章稍后會(huì)詳細(xì)討論的”繼承”,包含類間的父-子關(guān)系. 另外,PHP支持對(duì)屬性和方法的限制性訪問(wèn). 你可以聲明成員為private,不允許外部類訪問(wèn). 最后,PHP允許一個(gè)子類從它的父類中重載成員.
PHP5的對(duì)象模型把對(duì)象看成與任何其它數(shù)據(jù)類型不同,通過(guò)引用來(lái)傳遞. PHP不要求你通過(guò)引用(reference)顯性傳遞和返回對(duì)象. 在本章的最后將會(huì)詳細(xì)闡述基于引用的對(duì)象模型. 它是PHP5中最重要的新特性.
有了更直接的對(duì)象模型,就擁有了附加的優(yōu)勢(shì): 效率提高, 占用內(nèi)存少,并且具有更大的靈活性.
在PHP的前幾個(gè)版本中,腳本默認(rèn)復(fù)制對(duì)象.現(xiàn)在PHP5只移動(dòng)句柄,需要更少的時(shí)間. 腳本執(zhí)行效率的提升是由于避免了不必要的復(fù)制. 在對(duì)象體系帶來(lái)復(fù)雜性的同時(shí),也帶來(lái)了執(zhí)行效率上的收益. 同時(shí),減少?gòu)?fù)制意味著占用更少的內(nèi)存,可以留出更多內(nèi)存給其它操作,這也使效率提高.
Zand引擎2具有更大的靈活性. 一個(gè)令人高興的發(fā)展是允許析構(gòu)--在對(duì)象銷毀之前執(zhí)行一個(gè)類方法. 這對(duì)于利用內(nèi)存也很有好處,讓PHP清楚地知道什么時(shí)候沒(méi)有對(duì)象的引用,把空出的內(nèi)存分配到其它用途.
PHP5的內(nèi)存管理
對(duì)象傳遞
PHP5使用了Zend引擎II,對(duì)象被儲(chǔ)存于獨(dú)立的結(jié)構(gòu)Object Store中,而不像其它一般變量那樣儲(chǔ)存于Zval中(在PHP4中對(duì)象和一般變量一樣存儲(chǔ)于Zval)。在Zval中僅存儲(chǔ)對(duì)象的指針而不是內(nèi)容(value)。當(dāng)我們復(fù)制一個(gè)對(duì)象或者將一個(gè)對(duì)象當(dāng)作參數(shù)傳遞給一個(gè)函數(shù)時(shí),我們不需要復(fù)制數(shù)據(jù)。僅僅保持相同的對(duì)象指針并由另一個(gè)zval通知現(xiàn)在這個(gè)特定的對(duì)象指向的Object Store。由于對(duì)象本身位于Object Store,我們對(duì)它所作的任何改變將影響到所有持有該對(duì)象指針的zval結(jié)構(gòu)----表現(xiàn)在程序中就是目標(biāo)對(duì)象的任何改變都會(huì)影響到源對(duì)象。.這使PHP對(duì)象看起來(lái)就像總是通過(guò)引用(reference)來(lái)傳遞,因此PHP中對(duì)象默認(rèn)為通過(guò)“引用”傳遞,你不再需要像在PHP4中那樣使用&來(lái)聲明。
垃圾回收機(jī)制
某些語(yǔ)言,最典型的如C,需要你顯式地要求分配內(nèi)存當(dāng)你創(chuàng)建數(shù)據(jù)結(jié)構(gòu)。一旦你分配到內(nèi)存,就可以在變量中存儲(chǔ)信息。同時(shí)你也需要在結(jié)束使用變量時(shí)釋放內(nèi)存,這使機(jī)器可以空出內(nèi)存給其它變量,避免耗光內(nèi)存。
PHP可以自動(dòng)進(jìn)行內(nèi)存管理,清除不再需要的對(duì)象。PHP使用了引用計(jì)數(shù)(reference counting)這種單純的垃圾回收(garbage collection)機(jī)制。每個(gè)對(duì)象都內(nèi)含一個(gè)引用計(jì)數(shù)器,每個(gè)reference連接到對(duì)象,計(jì)數(shù)器加1。當(dāng)reference離開(kāi)生存空間或被設(shè)為NULL,計(jì)數(shù)器減1。當(dāng)某個(gè)對(duì)象的引用計(jì)數(shù)器為零時(shí),PHP知道你將不再需要使用這個(gè)對(duì)象,釋放其所占的內(nèi)存空間。
例如:
<?php class Person{} function sendEmailTo(){}
$haohappy = new Person( ); // 建立一個(gè)新對(duì)象: 引用計(jì)數(shù) Reference count = 1 $haohappy2 = $haohappy; // 通過(guò)引用復(fù)制: Reference count = 2 unset($haohappy); // 刪除一個(gè)引用: Reference count = 1 sendEmailTo($haohappy2); // 通過(guò)引用傳遞對(duì)象: // 在函數(shù)執(zhí)行期間: // Reference count = 2 // 執(zhí)行結(jié)束后: // Reference count = 1
unset($haohappy2); // 刪除引用: Reference count = 0 自動(dòng)釋放內(nèi)存空間
?>
