1.1.3.1.4 序列化再探討
從以上技術(shù)的討論中我們不難體會到,序列化是Java之所以能夠出色地實(shí)現(xiàn)其鼓吹的兩大賣點(diǎn)??分布式(distributed)和跨平臺(OS independent)的一個重要基礎(chǔ)。TIJ(即“Thinking in Java”)談到I/O系統(tǒng)時,把序列化稱為“l(fā)ightweight persistence”??“輕量級的持久化”,這確實(shí)很有意思。
★為什么叫做“序列”化?
開場白里我說更習(xí)慣于把“Serialization”稱為“序列化”而不是“串行化”,這是有原因的。介紹這個原因之前先回顧一些計算機(jī)基本的知識,我們知道現(xiàn)代計算機(jī)的內(nèi)存空間都是線性編址的(什么是“線性”知道吧,就是一個元素只有一個唯一的“前驅(qū)”和唯一的“后繼”,當(dāng)然頭尾元素是個例外;對于地址來說,它的下一個地址當(dāng)然不可能有兩個,否則就亂套了),“地址”這個概念推廣到數(shù)據(jù)結(jié)構(gòu),就相當(dāng)于“指針”,這個在本科低年級大概就知道了。注意了,既然是線性的,那“地址”就可以看作是內(nèi)存空間的“序號”,說明它的組織是有順序的,“序號”或者說“序列號”正是“Serialization”機(jī)制的一種體現(xiàn)。為什么這么說呢?譬如我們有兩個對象a和b,分別是類A和B的實(shí)例,它們都是可序列化的,而A和B都有一個類型為C的屬性,根據(jù)前面我們說過的原則,C當(dāng)然也必須是可序列化的。
1. import java.io.*;
2. ...
3. class A implements Serializable
4. {
5. C c;
6. ...
7. }
8.
9. class B implements Serializable
10. {
11. C c;
12. ...
13. }
14.
15. class C implements Serializable
16. {
17. ...
18. }
19.
20. A a;
21. B b;
22. C c1;
23. ...
注意,這里我們在實(shí)例化a和b的時候,有意讓他們的c屬性使用同一個C類型對象的引用,譬如c1,那么請試想一下,但我們序列化a和b的時候,它們的c屬性在外部字節(jié)流(當(dāng)然可以不僅僅是文件)里保存的是一份拷貝還是兩份拷貝呢?序列化在這里使用的是一種類似于“指針”的方案:它為每個被序列化的對象標(biāo)上一個“序列號”(serial number),但序列化一個對象的時候,如果其某個屬性對象是已經(jīng)被序列化的,那么這里只向輸出流寫入該屬性的序列號;從字節(jié)流恢復(fù)被序列化的對象時,也根據(jù)序列號找到對應(yīng)的流來恢復(fù)。這就是“序列化”名稱的由來!這里我們看到“序列化”和“指針”是極相似的,只不過“指針”是內(nèi)存空間的地址鏈,而序列化用的是外部流中的“序列號鏈”。
使用“序列號”而不是內(nèi)存地址來標(biāo)識一個被序列化的對象,是因?yàn)閺牧髦谢謴?fù)對象到內(nèi)存,其地址可能就未必是原來的地址了??我們需要的只是這些對象之間的引用關(guān)系,而不是死板的原始位置,這在RMI中就更是必要,在兩臺不同的機(jī)器之間傳遞對象(流),根本就不可能指望它們在兩臺機(jī)器上都具有相同的內(nèi)存地址。
相關(guān)推薦:計算機(jī)等級考試二級Java經(jīng)典算法大全匯總北京 | 天津 | 上海 | 江蘇 | 山東 |
安徽 | 浙江 | 江西 | 福建 | 深圳 |
廣東 | 河北 | 湖南 | 廣西 | 河南 |
海南 | 湖北 | 四川 | 重慶 | 云南 |
貴州 | 西藏 | 新疆 | 陜西 | 山西 |
寧夏 | 甘肅 | 青海 | 遼寧 | 吉林 |
黑龍江 | 內(nèi)蒙古 |