Java序列化和反序列化

一、簡介

Java序列化是將對象轉換為位元組序列的過程,以便在網絡上傳輸或保存到文件中。反序列化是將位元組序列轉換回對象的過程。它們是Java中非常重要的特性,可以幫助我們方便地將對象進行傳輸和保存,同時也是Java RMI(遠程方法調用)的基礎之一。

二、序列化實現

在Java中,我們可以將一個類序列化並保存到文件中,以便以後使用。這需要先實現 java.io.Serializable 接口,該接口沒有任何方法,只是一個標記接口。實現了 java.io.Serializable 接口的類才能被序列化。接着就可以使用 java.io.ObjectOutputStream 對象的 writeObject() 方法將對象序列化為位元組序列。

public class Student implements Serializable {
    private String name;
    private int age;
    private String address;

    public Student(String name, int age, String address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

    // getters and setters

    public static void main(String[] args) {
        Student student = new Student("張三", 18, "北京市");
        try {
            FileOutputStream fos = new FileOutputStream("student.dat");
            ObjectOutputStream oos = new ObjectOutputStream(fos);
            oos.writeObject(student);
            oos.close();
            fos.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

在上述示例中,我們實現了 Student 類並實現了 Serializable 接口。接下來,在 main 方法中創建了一個 Student 對象,然後將其寫入到名為 student.dat 的文件中。這個文件就是序列化後的位元組流。

三、反序列化實現

反序列化是將一個序列化的對象還原成原有的對象。需要使用 java.io.ObjectInputStream 對象的 readObject() 方法。在這之前,需要創建一個與序列化時相同的類,並且實現 Serializable 接口。

public static void main(String[] args) {
    try {
        FileInputStream fis = new FileInputStream("student.dat");
        ObjectInputStream ois = new ObjectInputStream(fis);
        Student student = (Student) ois.readObject();
        System.out.println(student.getName());
        ois.close();
        fis.close();
    } catch (IOException e) {
        e.printStackTrace();
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    }
}

在上述代碼中,我們首先讀取 student.dat 文件,然後使用 ObjectInputStream 對象的 readObject() 方法反序列化出一個Student 對象。最後打印該對象的姓名。

四、序列化UID的作用

Java序列化機制提供了一個叫做 serialVersionUID 的序列化版本號。這個版本號在序列化時會一起保存到文件中,反序列化時也會對比版本號是否一致,如果不一致,則會拋出一個InvalidClassException。

當一個類的實例被序列化時,serialVersionUID 的值也會被序列化保存下來。如果類的實現發生了變化(例如增加或刪除了字段),它的 serialVersionUID 可能會發生變化。因此,我們應該總是手動聲明 serialVersionUID,以確保正確性。

public class Student implements Serializable {
    private static final long serialVersionUID = 1L;

    private String name;
    private int age;
    private String address;

    // constructors and methods
}

五、transient關鍵字的作用

在Java中,有時候我們不想將某些字段序列化,這時候可以使用 transient 關鍵字。被 transient 修飾的字段會被忽略並跳過序列化過程。

public class Student implements Serializable {
    private String name;
    private int age;
    private transient String address;

    // constructors and methods
}

六、序列化的風險與預防措施

因為Java序列化會將對象的狀態以二進制數據的形式保存在磁盤上或在網絡中進行傳輸,因此可能會存在安全隱患,主要包括以下方面:

1. 反序列化漏洞:通過精心構造的序列化數據,黑客可以觸發執行任意的代碼。這個問題已在Java 8及以上版本中得到修復,但仍然需要對舊版本的代碼進行升級。

2. 替換可以序列化的類:如果需要序列化的類沒有明確指定 serialVersionUID,那麼在該類發生變化後(增加或刪除字段),反序列化時可能會導致程序崩潰。此時,黑客可以使用可序列化的替代類來執行攻擊。

3. 盜取會話信息:網站可能會將認證信息序列化後保存到Cookie中,如果黑客獲取了這個Cookie並解析出認證信息,就可以劫持用戶的會話。

為了解決上述問題,可以採取以下預防措施:

1. 明確指定 serialVersionUID:默認情況下,Java會通過計算類的哈希值來生成一個 serialVersionUID。因此,如果在類中增加或刪除變量,可能會導致 serialVersionUID 的變化,進而導致對象不能被反序列化。因此,建議明確指定 serialVersionUID 的值。

2. 使用白名單過濾類:在服務器端,可以對反序列化的類進行白名單過濾,只允許反序列化指定的類。

3. 針對特定場景開啟安全機制:對於需要保證安全的場景,可以開啟Java安全機制。例如,在Tomcat中可以開啟安全管理模塊,限制運行權限。

七、總結

Java序列化和反序列化是Java中非常重要的特性,可以幫助我們方便地將對象進行傳輸和保存。序列化時需要實現 java.io.Serializable 接口,並使用 java.io.ObjectOutputStream 對象將對象序列化為位元組序列;反序列化時需要使用 java.io.ObjectInputStream 對象將位元組序列反序列化為對象。為了保證安全性,需要手動指定 serialVersionUID,並在服務器端對反序列化的類進行白名單過濾。此外,也可以設置 transient 關鍵字來忽略某些字段的序列化,避免數據泄露。

原創文章,作者:INXJD,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/363893.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
INXJD的頭像INXJD
上一篇 2025-03-12 18:48
下一篇 2025-03-12 18:48

相關推薦

  • 金額選擇性序列化

    本文將從多個方面對金額選擇性序列化進行詳細闡述,包括其定義、使用場景、實現方法等。 一、定義 金額選擇性序列化指根據傳入的金額值,選擇是否進行序列化,以達到減少數據傳輸的目的。在實…

    編程 2025-04-29
  • java client.getacsresponse 編譯報錯解決方法

    java client.getacsresponse 編譯報錯是Java編程過程中常見的錯誤,常見的原因是代碼的語法錯誤、類庫依賴問題和編譯環境的配置問題。下面將從多個方面進行分析…

    編程 2025-04-29
  • Java JsonPath 效率優化指南

    本篇文章將深入探討Java JsonPath的效率問題,並提供一些優化方案。 一、JsonPath 簡介 JsonPath是一個可用於從JSON數據中獲取信息的庫。它提供了一種DS…

    編程 2025-04-29
  • Java Bean加載過程

    Java Bean加載過程涉及到類加載器、反射機制和Java虛擬機的執行過程。在本文中,將從這三個方面詳細闡述Java Bean加載的過程。 一、類加載器 類加載器是Java虛擬機…

    編程 2025-04-29
  • Java騰訊雲音視頻對接

    本文旨在從多個方面詳細闡述Java騰訊雲音視頻對接,提供完整的代碼示例。 一、騰訊雲音視頻介紹 騰訊雲音視頻服務(Cloud Tencent Real-Time Communica…

    編程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介紹

    本文將詳細介紹Java Milvus SearchParam withoutFields的相關知識和用法。 一、什麼是Java Milvus SearchParam without…

    編程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java語言中的一個版本,於2014年3月18日發佈。本文將從多個方面對Java 8中某一周的周一進行詳細的闡述。 一、數組處理 Java 8新特性之一是Stream…

    編程 2025-04-29
  • Java判斷字符串是否存在多個

    本文將從以下幾個方面詳細闡述如何使用Java判斷一個字符串中是否存在多個指定字符: 一、字符串遍歷 字符串是Java編程中非常重要的一種數據類型。要判斷字符串中是否存在多個指定字符…

    編程 2025-04-29
  • VSCode為什麼無法運行Java

    解答:VSCode無法運行Java是因為默認情況下,VSCode並沒有集成Java運行環境,需要手動添加Java運行環境或安裝相關插件才能實現Java代碼的編寫、調試和運行。 一、…

    編程 2025-04-29
  • Java任務下發回滾系統的設計與實現

    本文將介紹一個Java任務下發回滾系統的設計與實現。該系統可以用於執行複雜的任務,包括可回滾的任務,及時恢復任務失敗前的狀態。系統使用Java語言進行開發,可以支持多種類型的任務。…

    編程 2025-04-29

發表回復

登錄後才能評論