Java序列化之Serializable

来源:http://www.cnblogs.com/caoweizhao/archive/2016/10/31/6017447.html
-Advertisement-
Play Games

Java的序列化流程如下: Java的反序列化流程如下: 註意:並不是所有類都需要進行序列化,主要原因有兩個 1)安全問題。Java中有的類屬於敏感類,此類的對象數據不便對外公開,而序列化的對象數據很容易進行破解,無法保證其數據的安全性,因此一般這種類型的對象不會進行序列化。 2)資源問題。可以使用 ...


Java的序列化流程如下:

 

Java的反序列化流程如下:

註意:並不是所有類都需要進行序列化,主要原因有兩個

1)安全問題。Java中有的類屬於敏感類,此類的對象數據不便對外公開,而序列化的對象數據很容易進行破解,無法保證其數據的安全性,因此一般這種類型的對象不會進行序列化。

2)資源問題。可以使用序列化位元組流創建對象,而且這種創建時不受限制的,有時過多地創建對象會造成很大的資源問題,因此此類對象也不適宜進行序列化。

 

Serializable

Serializable是Java提供的一個序列化介面,它是一個空介面,專門為對象提供標準的序列化跟反序列化操作。

 

序列化過程:

1 Person p = new Person("name","id");
2         File file = new File("cache.txt");
3         FileOutputStream output = new FileOutputStream(file);
4         ObjectOutputStream objectOutputStream = new ObjectOutputStream(output);
5         objectOutputStream.writeObject(p);
6         output.close();
7         objectOutputStream.close();

反序列化過程:

1      File file = new File("cache.txt");
2         FileInputStream input= new FileInputStream(file);
3         ObjectInputStream objectInputStream = new ObjectInputStream(input);
4         Person p = (Person)objectInputStream.readObject();
5         System.out.println(p.getName()+"---"+p.getId());
6         input.close();
7         objectInputStream.close();

 

 

  • 需要序列化的類成員

    對象序列化時並不是所有成員都要轉換成二進位的位元組序列,因為為了節省存儲或傳輸空間以及提高序列化效率,有些不必要的成員是無需序列化的。其中包括:

    •   靜態變數。因為靜態變數屬於類的屬性,並不屬於某個具體實例,因此在序列化的時候無須進行序列化,反序列化時,可以直接獲取類的靜態成員引用。
    •   方法。方法只是一系列的操作集合,方法不會依賴對象,不會因為對象的不同,而操作不同,反序列化時,也可以從類中直接獲取方法信息。

 

  • 繼承關係的序列化
    • 父類實現Serializable時,子類被序列化,父類也會被序列化。
    • 父類沒有實現Serializable時,子類被序列化,父類不會被序列化

 

  • 引用關係的序列化

    如果對一個實現了Serializable的類進行序列化操作,則同時對它的引用類進行序列化操作。如果引用類沒有實現Serializable介面,JVM會拋出java.io.NotSerializableExeception.

1 class Person implements Serializable{
2     private String name;
3     private Tool tool = new Tool();
4 }
5 
6 class Tool implements Serializable{
7     
8 }

 

  此時對Person類進行序列化操作,則會同時對Tool類進行序列化操作。若Tool類沒有實現Serializable介面,則會拋出異常。

 

  • 保護敏感數據:
    •   一個類加上序列化標識後,該類對象的所有屬性信息將被序列化,然後進行本地存儲或網路傳輸。然後有時對象中的某些欄位屬於敏感信息,不應暴露出來。如果對其也進行序列化,容易被破解,從而   造成安全隱患,例如常見的密碼欄位。
    •   Java提供一個關鍵字transient,即瞬時關鍵字。該關鍵字關閉欄位的序列化,這樣受保護的信息就不會因為序列化而對外暴露。

 

  • 序列化標識ID
    •   試想一下這樣的情景:兩端進行網路傳輸序列化對象,由於某種原因,導致兩端使用的類的版本不同,假設接收方的類被刪除了幾個欄位。當發送發將對象的序列化位元組流發送到接收方時,由於接收方 的類少了幾個欄位,而無法解析。  
    •   Java要求實現序列化介面的類都必須聲明一個serialVersionUID靜態屬性,如果沒有該屬性JVM也會自動聲明該屬性,併為該屬性賦值(當類發生改變時會賦予不同的值)。該屬性的值是唯一的,用於 標識不同的序列化類。只有類的序列化標識完全相同,Java才會進行反序列化工作,這就是序列化標識的作用。
    •   對於前面提到的情景,假設沒有手動聲明serialVersionUID,則JVM對發送方跟接收方使用的類中的serialVersionUID賦予不同的值,則反序列化失敗。當手動給serialVersionUID賦值時,即使類的字 段發生改變,也能夠反序列化成功。

 

  • 自定義序列化策略
    •   定製序列化策略     

      Java提供了一套有效的機制,允許在序列化和反序列化時,使用定製的方法進行相應的處理。當傳輸雙方協定好序列化策略後,只需要在需要傳輸的序列化類中添加一組方法來實現這組策略,在序列化時會自動調用這些規定好的方法進行序列化和反序列化。方法如下:

     1)private void writeObject(ObjectOutputSteam out) throws IOException  

  在方法的內部有重要的代碼:out.defaultWriteObject()    //將對象數據以預設方式寫入到輸出流中

     2)private void readObject(ObjectInputStream in) throws IOException,ClassNotFoundException

  同樣的,此方法內部也有相似代碼:in.defaultReadObject();  //以預設方式從輸入流中恢復對象

這兩個方法的作用分別是將特定的對象寫入到輸出流中以及從輸入流中恢復特定的對象,通過這兩個方法,用戶即可實現自定義的序列化。當在實現Serializable介面的類中寫了上面兩個方法之後,序列化或反序列化該類時則會通過反射來調用這兩個方法,從而實現自定義序列化。

    •   限制序列化對象的數量     

      我們看下麵的單例模式:

 1 public class Singleton implements Serializable {
 2 
 3     private volatile static Singleton mInstance;
 4     private Singleton() {
 5     }
 6 
 7     public static Singleton getInstance() {
 8         if (mInstance == null) {
 9             synchronized (Singleton.class) {
10                 if (mInstance == null) {
11                     mInstance = new Singleton();
12                 }
13             }
14         }
15         return mInstance;
16     }
17 }

 

  此時通過反序列化獲取實例,則單例模式會失效。那該如何解決這個問題呢?

  Java有一種機制,可以讓我們在序列化和反序列化時,可以根據自己的需要,寫入或讀取指定的實例。使用這種機制,需要在實現Serializable介面的類中添加兩個方法:

  •     private Object readResolve()   //如果用戶在序列化類中添加了該方法,則在進行反序列化時,使用該方法返回的對象,作為反序列化對象。
  •     private Object writeReplace()   //如果用戶在序列化類中添加了該方法,則在進行序列化時,序列化該類返回的對象。

 

      再看使用了該機制的單例模式:

 1  1 public class Singleton implements Serializable {
 2  2 
 3  3     private volatile static Singleton mInstance;
 4  4     private Singleton() {
 5  5     }
 6  6 
 7  7     public static Singleton getInstance() {
 8  8         if (mInstance == null) {
 9  9             synchronized (Singleton.class) {
10 10                 if (mInstance == null) {
11 11                     mInstance = new Singleton();
12 12                 }
13 13             }
14 14         }
15 15         return mInstance;
16 16     }
17 17 
18 18     private Object readResolve() {
19 19         return getInstance();
20 20     }
21 21 
22 22     private Object writeReplace() {
23 23         return getInstance();
24 24     }
25 25 }

 

  此時的通過反序列化得到的對象也是同一個,即單例模式依然有效!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 首先我們新建一個項目,Java項目或者是Javaweb項目都可以,然後把hibernate需要的jar導入,我用的jar包如下: 然後新建一個實體類Teacher.java並添加get和set方法: 在Hibernate中一個實體類對應一個.hbm.xml文件,因此我們新建一個Teacher.hbm ...
  • Docker Introduction to Docker Monitoring Database MongoDB: The Good, The Bad, and The Ugly Web 4 Keys to a Clean Angular Implementation Vue.js server ... ...
  • 今日問題: 請問主程式輸出結果是?(點擊以下“【Java每日一題】20161101”查看20161031問題解析) 題目原發佈於公眾號、簡書:【Java每日一題】20161101,【Java每日一題】20161101 每日一題最新將在公眾號發佈,歡迎訂閱,交流進步 ...
  • 下麵是 Java 線程相關的熱門面試題,你可以用它來好好準備面試。 1) 什麼是線程? 線程是操作系統能夠進行運算調度的最小單位,它被包含在進程之中,是進程中的實際運作單位。程式員可以通過它進行多處理器編程,你可以使用多線程對運算密集型任務提速。比如,如果一個線程完成一個任務要 100 毫秒,那麼用 ...
  • 如今的互聯網,採集網站非常多,很多網站都喜歡盜鏈/盜用別人網站的圖片,這樣不僅侵犯網權,還導致被盜鏈的網站消耗大量的流量,給伺服器造成比較大的壓力,本文章向大家介紹php如何防止圖片盜用/盜鏈的兩種方法,需要的朋友可以參考一下。 圖片防盜鏈有什麼用? 防止其它網站盜用你的圖片,浪費你寶貴的流量。本文 ...
  • 這次主要是講解一下通過登錄後對得到的數據進行分頁,首先我們新建一個登錄頁面login.jsp,因為我們主要學習一下分頁,所以登錄驗證的部分不再闡述,主要代碼如下: 首先建立實體類User.java並添加get和set方法: 我們可以看到form表單是提交到pageServlet中,所以我們新建一個P ...
  • <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="http://user-20160821pd:8088/com.jdbc2/"> <title>My JSP 'inde ...
  • 搜索功能 DAO層都是一些資料庫的增刪改查操作 Servlet,控制層 點擊頁面的搜索,把輸入的信息提交到servlet, 實體Bean是針對資料庫中的欄位而建的, 不和資料庫做對應,而是打包一些零散的值的Bean,和它的頁面做對應,包名為:com.xxx.view 針對頁面的實體Bean 頁面亂碼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...