第四節--構造函數和析構函數
如果你在一個(gè)類(lèi)中聲明一個(gè)函數,命名為_(kāi)_construct,這個(gè)函數將被當成是一個(gè)構造函數并在建立一個(gè)對象實(shí)例時(shí)被執行. 清楚地說(shuō),__是兩個(gè)下劃線(xiàn). 就像其它任何函數一樣,構造函數可能有參數或者默認值. 你可以定義一個(gè)類(lèi)來(lái)建立一個(gè)對象并將其屬性全放在一個(gè)語(yǔ)句(statement)中.
你也可以定義一個(gè)名為_(kāi)_destruct的函數,PHP將在對象被銷(xiāo)毀前調用這個(gè)函數. 它稱(chēng)為析構函數.
繼承是類(lèi)的一個(gè)強大功能. 一個(gè)類(lèi)(子類(lèi)/派生類(lèi))可以繼承另一類(lèi)(父類(lèi)/基類(lèi))的功能. 派生類(lèi)將包含有基類(lèi)的所有屬性和方法,并可以在派生類(lèi)中加上其他屬性和方法. 你也可以覆寫(xiě)基類(lèi)的方法和屬性. 就像3.1.2中顯示的,你可以用extends關(guān)鍵字來(lái)繼承一個(gè)類(lèi).
你可能想知道構造函數是如何被繼承的. 當它們和其它方法一起被繼承時(shí),他們不會(huì )在創(chuàng )建對象時(shí)被執行.
如果你需要這個(gè)功能,你需要用第二章提到的::運算符. 它允許你指向一塊命名空間. parent指向父類(lèi)命名空間,你可以用parent::__construct來(lái)調用父類(lèi)的構造函數.
一些面向對象語(yǔ)言在類(lèi)之后命名構造函數. PHP的前幾個(gè)版本也是如此,到現在這種方法仍然有效.也就是:如果你把一個(gè)類(lèi)命名為Animal并且在其中建立一個(gè)命名也是Animal的方法,則這個(gè)方法就是構造函數.如果一個(gè)類(lèi)的同時(shí)擁有__construt構造函數和與類(lèi)名相同的函數,PHP將把__construct看作構造函數.這使得用以前的PHP版本所寫(xiě)的類(lèi)仍然可以使用. 但新的腳本(PHP5)應當使用__construct.
PHP的這種新的聲明構造函數的方法可以使構造函數有一個(gè)獨一無(wú)二的名稱(chēng),無(wú)論它所在的類(lèi)的名稱(chēng)是什么. 這樣你在改變類(lèi)的名稱(chēng)時(shí),就不需要改變構造函數的名稱(chēng).
你可能在PHP中給構造函數一個(gè)像其它類(lèi)方法一樣的訪(fǎng)問(wèn)方式. 訪(fǎng)問(wèn)方式將會(huì )影響從一定范圍內實(shí)例化對象的能力. 這允許實(shí)現一些固定的設計模式,如Singleton模式.
析構函數,相反于構造函數. PHP調用它們來(lái)將一個(gè)對象從內存中銷(xiāo)毀. 默認地,PHP僅僅釋放對象屬性所占用的內存并銷(xiāo)毀對象相關(guān)的資源. 析構函數允許你在使用一個(gè)對象之后執行任意代碼來(lái)清除內存.
當PHP決定你的腳本不再與對象相關(guān)時(shí),析構函數將被調用. 在一個(gè)函數的命名空間內,這會(huì )發(fā)生在函數return的時(shí)候. 對于全局變量,這發(fā)生于腳本結束的時(shí)候. 如果你想明確地銷(xiāo)毀一個(gè)對象,你可以給指向該對象的變量分配任何其它值. 通常將變量賦值勤為NULL或者調用unset .
下面的例子中,計算從類(lèi)中實(shí)例化的對象的個(gè)數. Counter類(lèi)從構造函數開(kāi)始增值,在析構函數減值.
一旦你定義了一個(gè)類(lèi),你可以用new來(lái)建立一個(gè)這個(gè)類(lèi)的實(shí)例. 類(lèi)的定義是設計圖,實(shí)例則是放在裝配線(xiàn)上的元件. New需要類(lèi)的名稱(chēng),并返回該類(lèi)的一個(gè)實(shí)例. 如果構造函數需要參數,你應當在new后輸入參數.
class Counter
{
private static $count = 0;
function __construct()
{
self::$count++;
}
function __destruct()
{
self::$count--;
}
function getCount()
{
return self::$count;
}
}
//建立第一個(gè)實(shí)例
$c = new Counter();
//輸出1
print($c->getCount() . "
n");
//建立第二個(gè)實(shí)例
$c2 = new Counter();
//輸出2
print($c->getCount() . "
n");
//銷(xiāo)毀實(shí)例
$c2 = NULL;
//輸出1
print($c->getCount() . "
n");
?>
當你新建了一個(gè)實(shí)例,內存會(huì )被準備來(lái)存儲所有屬性. 每個(gè)實(shí)例有自己獨有的一組屬性. 但方法是由該類(lèi)的所有實(shí)例共享的.