2011年10月16日 星期日

[ Java 文章收集 ] Java Serializable(序列化)的理解和總結


轉載自 這裡
前言 :
序列化過程簡單說就是為了保存在內存中的各種對象的狀態(也就是實例變量,不只是類別本身),並且可以把保存的對象狀態再讀出來。雖然你可以用你自己的各種各樣的方法來保存object states,但是Java給你提供一種應該比你自己好的保存對象狀態的機制,那就是序列化.

什麽情況下需要序列化 :
* 當你想把的內存中的對象狀態保存到一個文件中或者數據庫中時候;
* 當你想用套接字在網絡上傳送對象的時候;
* 當你想通過RMI傳輸對象的時候

在沒有序列化前,每個保存在堆(Heap)中的對象都有相應的狀態(state),即實例變量(instance ariable)比如 :
  1. Foo  myFoo = new Foo();    
  2. myFoo .setWidth(37);    
  3. myFoo.setHeight(70);  
當 通過下面的代碼序列化之後,MyFoo對象中的width和Height實例變量的值(37,70)都被保存到foo.ser文件中,這樣以後又可以把它 從文件中讀出來,重新在堆中創建原來的對象。當然保存時候不僅僅是保存對象的實例變量的值,JVM還要保存一些小量信息,比如類的類型等以便恢復原來的對 象.

實現序列化的步驟 : (保存到一個文件)
- Make a FileOutputStream
  1. FileOutputStream fs = new FileOutputStream("foo.ser");  
- Make a ObjectOutputStream
  1. ObjectOutputStream os =  new ObjectOutputStream(fs);  
- Write the object
  1. os.writeObject(myObject1);    
  2. os.writeObject(myObject2);    
  3. os.writeObject(myObject3);  
- Close the ObjectOutputStream 
簡單範例 :
  1. import java.io.*;  
  2.     
  3. public class  Box implements Serializable    
  4. {    
  5.     private int width;    
  6.     private int height;    
  7.     
  8.     public void setWidth(int width){    
  9.         this.width  = width;    
  10.     }    
  11.     public void setHeight(int height){    
  12.         this.height = height;    
  13.     }    
  14.     
  15.     public static void main(String[] args){    
  16.         Box myBox = new Box();    
  17.         myBox.setWidth(50);    
  18.         myBox.setHeight(30);    
  19.     
  20.         try{    
  21.             FileOutputStream fs = new FileOutputStream("foo.ser");    
  22.             ObjectOutputStream os =  new ObjectOutputStream(fs);    
  23.             os.writeObject(myBox);    
  24.             os.close();    
  25.         }catch(Exception ex){    
  26.             ex.printStackTrace();    
  27.         }    
  28.     }      
  29. }  
相關註意事項 :
* 序列化時,只對對象的狀態進行保存,而忽略對象的方法.
* 當一個父類實現序列化,子類自動實現序列化,不需要顯式實現Serializable接口.
* 當一個對象的實例變量引用其他對象,序列化該對象時也把引用對象進行序列化
* 並非所有的對象都可以序列化,,至於為什麽不可以,有很多原因了,比如 :
1.安全方面的原因,比如一個對象擁有private,public等field,對於一個要傳輸的對象,比如寫到文件,或者進行rmi傳輸 等等,在序列化進行傳輸的過程中,這個對象的private等域是不受保護的。
2. 資源分配方面的原因,比如socket,thread類,如果可以序列化,進行傳輸或者保存,也無法對他們進行重新的資源分 配,而且,也是沒有必要這樣實現.

補充說明 :
Tutorials Point : Java - Serialization
Java provides a mechanism, called object serialization where an object can be represented as a sequence of bytes that includes the object's data as well as information about the object's type and the types of data stored in the object...

[ Gossip in Java(2) ] 輸入輸出 : 位元串流 (ObjectInputStream、ObjectOutputStream)
在Java這樣支援物件導向的程式中撰寫程式,很多資料都是以物件的方式存在,在程式運行過後,您會希望將這些資料加以儲存,以供下次執行程式時使用,這時您可以使用ObjectInputStream、ObjectOutputStream 來進行這項工作...

2 則留言:

  1. 我得到java.io.NotSerializableException的錯誤ㄟ
    Main.log.info = "CNPC.saveProject()";
    try{
    ArrayList projectList = Main.loadProject();
    projectList.add(new Main.projectItem(
    name,
    path
    ));
    FileOutputStream fos = new FileOutputStream(Main.projectListFilePath);
    Main.log.log(Main.projectListFilePath);
    ObjectOutputStream oos = new ObjectOutputStream(fos);
    Main.log.cp("ready to save projectItem to file!");
    oos.writeObject(projectList);
    oos.close();
    fos.close();
    Main.log.cp("File closed!");
    }catch(Exception e){
    JOptionPane.showMessageDialog(new JFrame(), e, "Error", JOptionPane.ERROR_MESSAGE);
    Main.log.log(e.toString());
    }
    Main.log.info = "";


    可以幫幫我嗎?

    回覆刪除
    回覆
    1. Check below link:
      https://stackoverflow.com/questions/13895867/java-io-notserializableexception

      刪除

[Git 常見問題] error: The following untracked working tree files would be overwritten by merge

  Source From  Here 方案1: // x -----删除忽略文件已经对 git 来说不识别的文件 // d -----删除未被添加到 git 的路径中的文件 // f -----强制运行 #   git clean -d -fx 方案2: 今天在服务器上  gi...