Java序列化和反序列化示例介紹

以前用序列化都是一些方法需要才實現的,後來業務需求要深拷貝才去研究。參閱瞭別人博客得出一些總結。

序列化是為瞭把Java對象轉化為字節序列(字節流)的過程。然後深拷貝是通過對流的操作來實現的,序列化後數據方便存儲傳輸。反序列化則是把字節序列反序列化為Java對象

存儲方便:因為對象會被回收,序列化後可以持續化存儲在磁盤中
傳輸方便:字節序列(二進制形式)可以進行網絡傳輸和傳播。

最好設置一個SerialversionUID,因為序列化和反序列化是對比SerialversionUID來進行的,雖然不設置接口也會默認生成一個,但是要知道序列化對象過程一般都是對象->序列化->存儲或傳輸->反序列化。
舉個例子:
先創建一個實體類Student

import lombok.Data;

import java.io.Serializable;

@Data
public class Student implements Serializable {
    private Integer id;
    private String name;
    private String sex;
}
然後創建一個測試類SerializableTest
import serialization.entity.Student;

import java.io.*;

public class SerializableTest {
    public static void main(String[] args) throws Exception {
        serializeStudent();
        Student student = deserializeStudent();
        System.out.println("name:" + student.getName());
        System.out.println("sex:" + student.getSex());
    }

    private static void serializeStudent() throws IOException {
        Student student = new Student();
        student.setId(1);
        student.setName("張三");
        student.setSex("male");

        ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(
                new File("F:/student.txt")));
        out.writeObject(student);
        System.out.println("序列化成功");
        out.close();
    }

    private static Student deserializeStudent() throws Exception {
        ObjectInputStream in = new ObjectInputStream(new FileInputStream(new File("F:/student.txt")));
        Student student = (Student) in.readObject();
        System.out.println("反序列化成功");
        return student;
    }
}
執行結果:
序列化成功
反序列化成功
name:張三
sex:male
這個時候沒有指定SerialversionUID也是可以成功的,但對象->序列化->存儲或傳輸->反序列化,咱們在反序列化操作之前對Student類修改呢?

這個時候咱們修改一下代碼,先註釋掉反序列化代碼,先進行序列化。

import serialization.entity.Student;

import java.io.*;

public class SerializableTest {
    public static void main(String[] args) throws Exception {
        serializeStudent();
//        Student student = deserializeStudent();
//        System.out.println("name:" + student.getName());
//        System.out.println("sex:" + student.getSex());
    }

    private static void serializeStudent() throws IOException {
        Student student = new Student();
        student.setId(1);
        student.setName("張三");
        student.setSex("male");

        ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(
                new File("F:/student.txt")));
        out.writeObject(student);
        System.out.println("序列化成功");
        out.close();
    }

//    private static Student deserializeStudent() throws Exception {
//        ObjectInputStream in = new ObjectInputStream(new FileInputStream(new File("F:/student.txt")));
//        Student student = (Student) in.readObject();
//        System.out.println("反序列化成功");
//        return student;
//    }
}
運行結果:
序列化成功

修改Student類

import lombok.Data;

import java.io.Serializable;

@Data
public class Student implements Serializable {
    private Integer id;
    private String name;
    private String sex;
    private String address;
}
註釋掉序列化方法,進行反序列化
import serialization.entity.Student;

import java.io.*;

public class SerializableTest {
    public static void main(String[] args) throws Exception {
//        serializeStudent();
        Student student = deserializeStudent();
        System.out.println("name:" + student.getName());
        System.out.println("sex:" + student.getSex());
    }

//    private static void serializeStudent() throws IOException {
//        Student student = new Student();
//        student.setId(1);
//        student.setName("張三");
//        student.setSex("male");
//
//        ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(
//                new File("F:/student.txt")));
//        out.writeObject(student);
//        System.out.println("序列化成功");
//        out.close();
//    }

    private static Student deserializeStudent() throws Exception {
        ObjectInputStream in = new ObjectInputStream(new FileInputStream(new File("F:/student.txt")));
        Student student = (Student) in.readObject();
        System.out.println("反序列化成功");
        return student;
    }
}
執行結果:
Exception in thread "main" java.io.InvalidClassException: serialization.entity.Student; local class incompatible: stream classdesc serialVersionUID = 3846952599709361171, local class serialVersionUID = -4606152942663467236
	at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:699)
	at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1885)
	at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1751)
	at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:2042)
	at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1573)
	at java.io.ObjectInputStream.readObject(ObjectInputStream.java:431)
	at serialization.demo.SerializableTest.deserializeStudent(SerializableTest.java:30)
	at serialization.demo.SerializableTest.main(SerializableTest.java:10)

Process finished with exit code 1
可以看出兩次的執行的SerialversionUID不匹配,導致產生java.io.InvalidClassException異常,所以隻要指定瞭SerialversionUID就不會報異常。
//指定serialVersionUID正確寫法
private static final long serialVersionUID = 3846952599709361171L;

//如果已經進行序列化瞭不知道SerialversionUID,可以通過反射獲取
Object obj = Student.class.newInstance();
Field field = Student.class.getDeclaredField("serialVersionUID");
field.setAccessible(true);
System.out.println(field.getLong(obj));

最後需要知道的一點就是字節流字符流的區別。
字節流:傳輸過程中,傳輸數據的最基本單位是字節的流。
字符流:傳輸過程中,傳輸數據的最基本單位是字符的流。

這樣講可能有點不知所雲,字節其實就是Java的八大基本類型Byte(比特)單位,而字符通常是’A’、‘B’、’$’、’&'等,字節大小則取決於你是什麼編碼(環境),如下:

ASCII 碼中,一個英文字母(不分大小寫)為一個字節,一個中文漢字為兩個字節。

UTF-8 編碼中,一個英文字為一個字節,一個中文為三個字節。

Unicode 編碼中,一個英文為一個字節,一個中文為兩個字節。

符號:英文標點為一個字節,中文標點為兩個字節。例如:英文句號 . 占1個字節的大小,中文句號 。

占2個字節的大小。UTF-16 編碼中,一個英文字母字符或一個漢字字符存儲都需要 2 個字節(Unicode 擴展區的一些漢字存儲需要 4 個字節)。

UTF-32 編碼中,世界上任何字符的存儲都需要 4 個字節。

到此這篇關於Java序列化和反序列化示例介紹的文章就介紹到這瞭,更多相關Java序列化內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: