|
所謂對象序列化就是將對象的狀態(tài)轉(zhuǎn)換成字節(jié)流,以后可以通過這些值再生成相同狀態(tài)的對象。這個過程也可以通過網(wǎng)絡(luò)實(shí)現(xiàn),可以先在Windows機(jī)器上創(chuàng)建一個對象,對其序列化,然后通過網(wǎng)絡(luò)發(fā)給一臺Unix機(jī)器,然后在那里準(zhǔn)確無誤地重新“裝配”。是不是很神奇。 也許你會說,只了解一點(diǎn)點(diǎn),但從來沒有接觸過,其實(shí)未必如此。RMI、Socket、JMS、EJB你總該用過一種吧,彼此為什么能夠傳遞Java對象,當(dāng)然都是對象序列化機(jī)制的功勞。 第一次使用Java的對象序列化是做某項(xiàng)目,當(dāng)時要求把幾棵非常復(fù)雜的樹(JTree)及相應(yīng)的數(shù)據(jù)保存下來(就是我們常用的保存功能),以便下次運(yùn)行程序時可以繼續(xù)上次的操作。 那時XML技術(shù)在網(wǎng)上非常的熱,而且功能也強(qiáng)大,再加上樹的結(jié)構(gòu)本來就和XML存儲數(shù)據(jù)的格式很像。作為一項(xiàng)對新技術(shù)比較有興趣的我當(dāng)然很想嘗試一下。不過經(jīng)過仔細(xì)分析,發(fā)現(xiàn)如果采用XML保存數(shù)據(jù),后果真是難以想象:哪棵樹的哪個節(jié)點(diǎn)被展開、展開到第幾級、節(jié)點(diǎn)當(dāng)前的屬性是什么。真是不知該用A、B、C還是用1、2、3來表示。 還好,發(fā)現(xiàn)了Java的對象序列化機(jī)制,問題迎刃而解,只需簡單的將每棵樹的根節(jié)點(diǎn)序列化保存到硬盤上,下次再通過反序列化后的根節(jié)點(diǎn)就可以輕松的構(gòu)造出和原來一模一樣的樹來。 其實(shí)保存數(shù)據(jù),尤其是復(fù)雜數(shù)據(jù)的保存正是對象序列化的典型應(yīng)用。最近另一個項(xiàng)目就遇到了需要對非常復(fù)雜的數(shù)據(jù)進(jìn)行存取,通過使用對象的序列化,問題同樣化難為簡。 對象的序列化還有另一個容易被大家忽略的功能就是對象復(fù)制(Clone),Java中通過Clone機(jī)制可以復(fù)制大部分的對象,但是眾所周知,Clone有深層Clone和淺層Clone,如果你的對象非常非常復(fù)雜,假設(shè)有個100層的Collection(夸張了點(diǎn)),如果你想實(shí)現(xiàn)深層Clone,真是不敢想象,如果使用序列化,不會超過10行代碼就可以解決。 還有就是Swing組件,如果你有兩個很象很象(或是一模一樣)的比較難以構(gòu)造的Swing組件,你該怎么辦,也許你想到了Clone,但是偏偏Java的Swing組件沒有提供Clone方法。別急,使用序列化,6行代碼搞定:
雖然Java的序列化非常簡單、強(qiáng)大,但是要用好,還有很多地方需要注意。比如曾經(jīng)序列化了一個對象,可由于某種原因,該類做了一點(diǎn)點(diǎn)改動,然后重新被編譯,那么這時反序列化剛才的對象,將會出現(xiàn)異常。 你可以通過添加serialVersionUID屬性來解決這個問題。如果你的類是個單態(tài)(Singleton)類,是否允許用戶通過序列化機(jī)制復(fù)制該類,如果不允許你需要謹(jǐn)慎對待該類的實(shí)現(xiàn)。 |
|
|