Java學習路線:Java中的對象流和序列化,最近,在講流的使用,其中對象流的作用其實就是將自定義類的對象與流之間相互轉(zhuǎn)換的流。 看起來還是挺簡單的,那么看下面的例子: public class Student{ private int id; private String name; private String sex; private String tel; // 此處省略getter,setter public Student(int id, String name, String sex, String tel) { super(); this.id = id; this.name = name; this.sex = sex; this.tel = tel; } } public class TestObjectStream { public static void main(String[] args) { writeObject(); readObject(); } public static void writeObject() { try { FileOutputStream fos = new FileOutputStream("C:/Users/wangliang/Desktop/1"); ObjectOutputStream oos = new ObjectOutputStream(fos); Student student1 = new Student(1, "Mary", "female", "11111"); Student student2 = new Student(2, "Jack", "male", "2222"); Student student3 = new Student(3, "Andy", "male", "3333"); List<Student> list = Arrays.asList(student1, student2, student3); oos.writeObject(list); // 寫入對象 oos.flush(); fos.close(); oos.close(); } catch (Exception e) { e.printStackTrace(); } } public static void readObject() { try { FileInputStream fis = new FileInputStream("C:/Users/wangliang/Desktop/1"); ObjectInputStream ois = new ObjectInputStream(fis); List<Student> list = (List<Student>)ois.readObject(); // 讀取對象 list.forEach(System.out::println); fis.close(); ois.close(); } catch (Exception e) { e.printStackTrace(); } } } 然后在運行的過程發(fā)現(xiàn)出現(xiàn)如下的異常: java.io.NotSerializableException: com.qianfeng.day33.entity.Student at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1184) at java.io.ObjectOutputStream.writeArray(ObjectOutputStream.java:1378) at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1174) at java.io.ObjectOutputStream.defaultWriteFields(ObjectOutputStream.java:1548) at java.io.ObjectOutputStream.writeSerialData(ObjectOutputStream.java:1509) at java.io.ObjectOutputStream.writeOrdinaryObject(ObjectOutputStream.java:1432) at java.io.ObjectOutputStream.writeObject0(ObjectOutputStream.java:1178) at java.io.ObjectOutputStream.writeObject(ObjectOutputStream.java:348) at com.qianfeng.day33.test.TestObjectStream.writeObject(TestObjectStream.java:28) at com.qianfeng.day33.test.TestObjectStream.main(TestObjectStream.java:16) 根據(jù)此異常的說明告訴我們,該實體類沒有序列化,加上序列化的代碼試試,如下: public class Student implements java.io.Serializable{ private static final long serialVersionUID = 4500674287188895027L; private int id; private String name; private String sex; private String tel; // 此處省略getter,setter public Student(int id, String name, String sex, String tel) { super(); this.id = id; this.name = name; this.sex = sex; this.tel = tel; } } 就是實現(xiàn)了一個java.io.Serializable接口,但是查看一下該接口源代碼的說明: * * @author unascribed * @see java.io.ObjectOutputStream * @see java.io.ObjectInputStream * @see java.io.ObjectOutput * @see java.io.ObjectInput * @see java.io.Externalizable * @since JDK1.1 */ public interface Serializable { } 發(fā)現(xiàn)該接口中并沒有任何內(nèi)容,上面描述中@see后面幾乎就是對象流的一些操作,說明序列化的作用與對象流是相關(guān)的,那么序列化到底干了什么呢? 我們可以通過觸類旁通的方式,來看看其他的語言中對序列化的要求,比如OC這門語言中的要求如下: //流的數(shù)據(jù)(JSON數(shù)據(jù)) ---->OC對象 + (id)JSONObjectWithData:(NSData *)data options:(NSJSONReadingOptions)opt error:(NSError **)error; //OC對象 ------->流的數(shù)據(jù)(Json數(shù)據(jù)) + (NSData *)dataWithJSONObject:(id)obj options:(NSJSONWritingOptions)opt error:(NSError **)error; 在OC中,如果一個實體類需要實現(xiàn)序列化,必須要實現(xiàn)這兩個方法,這兩個方法的作用是告知系統(tǒng),當需要將實體類以流的形式傳輸或保存成文件時,該如何轉(zhuǎn)換成一個JSON數(shù)據(jù)(指定下key和value),然后可以使用字符串的方式轉(zhuǎn)換成二進制(byte數(shù)組)。 那么在Java中為什么不需要呢?因為Java中有反射機制,可以很容易的完成這個內(nèi)容。 那么序列化中序列號的作用是什么呢?是為了保證序列化和反序列化能夠正確進行,只有當序列號一致,才能轉(zhuǎn)換成功。 |
|
來自: 好程序員IT > 《Java培訓教程》