序列化(Serializing)
PHP不支持永久對象,在OOP中永久對象是可以在多個應用的引用中保持狀態和功能的對象,這意味著擁有將對象保存到一個文件或數據庫中的能力,而 且可以在以後裝入對象。這就是所謂的序列化機制。PHP 擁有序列化方法,它可以通過對象進行調用,序列化方法可以返回對象的字符串表示。然而,序列化只保存了對象的成員數據而不包話方法。
在 PHP4中,如果你將對象序列化到字符串$s中,然後釋放對象,接著反序列化對象
到$obj,你可以繼續使用對象的方法!我不建議這樣去做,因為(a)文 檔中沒有保證這種行為在以後的版本中仍然可以使用。(b) 這個可能導致一種誤解,在你把一個序列化後的版本保存到磁盤並退出腳本時。當以後運行這個腳本時,你不能期待著在反序列化一個對象時,對象的方法也會在那 裡,因為字符串表示根本就不包括方法。
總而言之,PHP 進行序列化對於保存對象的成員變量非常有用。(你也可以將相關數組和數組序列化到一個文件中)。
例子:
復制代碼 代碼如下:
<?php
$obj=new Classfoo();
$str=serialize($obj);
//保存$str到磁盤上
//幾個月以後
//從磁盤中裝入str
$obj2=unserialize($str)
你恢復了成員數據,但是不包括方法(根據文檔所說)。這導致了只能通過類似於使用$obj2->x來存取成員變量(你沒有別的方法!)的唯一辦法,所以不要在家裡試它。
有一些辦法可以解決這個問題,我把它留著,因為對這篇簡潔的文章來說,他們太不好。我會很高興地歡迎在PHP的後續版本中有全序列化的特性。
使用類進行數據存儲PHP和OOP一件非常好的事情就是,你可以很容易地定義一個類來操作某件事情,並且無論何時你想用的時候都可以調用相應的類。 假設你有一個HTML表單,用戶可以通過選擇產品ID號來選擇一個產品。在數據庫中有產品的信息,你想把產品顯示出來,顯示它的價格等等。你擁有不同類型 的產品,並且同一個動作可能對不同的產品具有不同的意思。例如,顯示一個聲音可能意味著播放它,但是對於其它種類的產品可能意味著顯示一個存在數據庫中的 圖片。你可以使用OOP或PHP來減少編碼並提高質量。
定義一個產品的類,定義它應該有的方法(例如:顯示),然後定義對每一種類型的產品
的類,從產品類派後出來(SoundItem類,ViewableItem類,等等),覆蓋在產品類中的方法,使它們按你的想法動作。
根據數據庫中每一種產品的類型(type)字段給類命名,一個典型的產品表可能有(id, type, price, description, 等等字段)...然後在處理腳本中,你可以從數據庫中取出type值,然後實例化一個名為type的對象:
復制代碼 代碼如下:
<?php
$obj=new $type();
$obj->action();
這是PHP的一個非常好的特性,你可以不用考慮對象的類型,調用$obj的顯示方法或其它的方法。使用這個技術,你不需要修改腳本去增加一個新類型的對象,只是增加一個處理它的類。
這個功能很強大,只要定義方法,而不去考慮所有對象的類型,在不同的類中按不同的方法實現它們,然後在主腳本中對任意對象使用它們,沒有if...else,也不需要兩個程序員,
只有高興。
現在你同意編程是容易的,維護是便宜的,可重用是真的嗎?
如果你管理一組程序員,分配工作就是很簡單的了,每個人可能負責一個類型的對象和
處理它的類。
可以通過這個技術實現國際化,根據用戶所選的語言字段應用相應的類就可以了,等等。
拷貝和克隆
當你創建一個$obj的對象時,你可以通過$obj2=$obj來拷貝對象,新的對 象是$obj的一個拷貝(不是一個引用),所以它具有$obj在當時的狀態。有時候,你不想這樣,你只是想生成一個象obj類一樣的一個新的對象,可以通 過使用new語句來調用類的構造函數。在PHP中也可以通過序列化,和一個基類來實現,但所有的其它類都要從基類派生出來。
進入危險區域
當你序列化一個對象,你會得到某種格式的字符串,如果你感興趣,你可以調究它,其中,字符串中有類的名字(太好了!),你可以把它取出來,象:
復制代碼 代碼如下:
<?php
$herring=serialize($obj);
$vec=explode(':',$herring);
$nam=str_replace("\"",'',$vec[2]);
所以假設你創建了一個"Universe"的類,並且強制所有的類都必須從universe擴展,你可以在universe 中定義一個clone的方法,如下:
復制代碼 代碼如下:
<?php
function clone() {
$herring=serialize($this);
$vec=explode(':',$herring);
$nam=str_replace("\"",'',$vec[2]);
$ret=new $nam;
return $ret;
}
}
//然後
$obj=new Something();
//從Universe擴展
$other=$obj->clone();
你所得到的是一個新的Something類的對象,它同使用new方法,調用構造函數創建出的對象一樣。我不知道這個對你是否有用,但是Universe類可以知道派生類的名字是一個好的經驗。想象是唯一的限制。
這個語句就是寫入當前的時間。