面試官:Java序列化為什麼要實現Serializable介面?我懵了

来源:https://www.cnblogs.com/chengxy-nds/archive/2020/02/22/12347324.html
-Advertisement-
Play Games

整理了一些Java方面的架構、面試資料(微服務、集群、分散式、中間件等),有需要的小伙伴可以關註公眾號【程式員內點事】,無套路自行領取 更多優選 "一口氣說出 9種 分散式ID生成方式,面試官有點懵了" "3萬字總結,Mysql優化之精髓" "為了不複製粘貼,我被逼著學會了JAVA爬蟲" "技術部突 ...


整理了一些Java方面的架構、面試資料(微服務、集群、分散式、中間件等),有需要的小伙伴可以關註公眾號【程式員內點事】,無套路自行領取

更多優選

寫在前邊

最近有個公眾號粉絲和我聊了聊他面試的經歷,一個剛入坑Java兩年的新人,由於疫情原因視頻面試,而面試官只問了一個問題:“Java序列化為什麼要實現Serializable介面?”,結果他一時語塞面試OVER。說實話聽到這個問題,我也有些懵逼,平時忙著研究各種中間件、什麼高可用框架,可真要回頭對Java基礎知識較起真,發現自己的技術債欠的太多,所以和大家一起複習一下Java序列化知識。


什麼是Java序列化?

序列化Java中的序列化機制能夠將一個實例對象信息寫入到一個位元組流中(只序列化對象的屬性值,而不會去序列化方法),序列化後的對象可用於網路傳輸,或者持久化到資料庫、磁碟中。

反序列化:需要對象的時候,再通過位元組流中的信息來重構一個相同的對象。

Java中要使一個類可以序列化,實現java.io.Serializable介面是最簡單的。

public class User implements Serializable {

    private static final long serialVersionUID = 1L;
}

那麼我們來看看Serializable介面的源碼實現,可以看到Serializable介面中並沒有方法或欄位,這個介面僅僅用於標識可序列化的語義,也就是說它只是用來標識一個對象是否可被序列化。

package java.io;

/**
 * @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 {
}

接下來寫一個對象信息寫入磁碟的例子測試一下:

創建一個User對象,並實現Serializable介面

@Data
public class User implements Serializable {

    private static final long serialVersionUID = 1L;

    private String name;

    private String age;
}

User對象信息寫入到磁碟當中

@Slf4j
public class serializeTest {
    
    public static void main(String[] args) throws Exception {
        User user = new User();
        user.setName("fufu");
        user.setAge("18");

        serialize(user);
        log.info("Java序列化前的結果:{} ", user);

        User duser = deserialize();
        log.info("Java反序列化的結果:{} ", duser);
    }
    /**
     * @author xzf
     * @description 序列化
     * @date 2020/2/22 19:34
     */
    private static void serialize(User user) throws Exception {
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(new File("D:\\111.txt")));
        oos.writeObject(user);
        oos.close();
    }
    /**
     * @author xzf
     * @description 反序列化
     * @date 2020/2/22 19:34
     */
    private static User deserialize() throws Exception {
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream(new File("D:\\111.txt")));
        return (User) ois.readObject();
    }
}
序列化前的結果: User(name=fufu, age=18)
反序列化後的結果: User(name=fufu, age=18)

打開writeObject方法的源碼看一下,發現方法中有這麼一個邏輯,當要寫入的對象是StringArrayEnumSerializable類型的對象則可以正常序列化,否則會拋出NotSerializableException異常。

這就能解釋為什麼Java序列化一定要實現Serializable介面了。

/**
     * Underlying writeObject/writeUnshared implementation.
     */
    private void writeObject0(Object obj, boolean unshared)
        throws IOException
    {
        boolean oldMode = bout.setBlockDataMode(false);
        depth++;
        try {
            // 省略號。。。。。。。。。。

            // remaining cases
            if (obj instanceof String) {
                writeString((String) obj, unshared);
            } else if (cl.isArray()) {
                writeArray(obj, desc, unshared);
            } else if (obj instanceof Enum) {
                writeEnum((Enum<?>) obj, desc, unshared);
            } else if (obj instanceof Serializable) {
                writeOrdinaryObject(obj, desc, unshared);
            } else {
                if (extendedDebugInfo) {
                    throw new NotSerializableException(
                        cl.getName() + "\n" + debugInfoStack.toString());
                } else {
                    throw new NotSerializableException(cl.getName());
                }
            }
        } finally {
            depth--;
            bout.setBlockDataMode(oldMode);
        }
    }

那麼可能會有人疑問,String為啥就不用實現Serializable介面呢?其實String已經內部實現了Serializable,不用我們再顯示實現。看看源碼就懂了

public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];

    /** Cache the hash code for the string */
    private int hash; // Default to 0

    /** use serialVersionUID from JDK 1.0.2 for interoperability */
    private static final long serialVersionUID = -6849794470754667710L;

    ......
}

既然已經實現了Serializable介面,為什麼還要顯示指定serialVersionUID的值呢?

因為序列化對象時,如果不顯示的設置serialVersionUID,Java在序列化時會根據對象屬性自動的生成一個serialVersionUID,再進行存儲或用作網路傳輸。

在反序列化時,會根據對象屬性自動再生成一個新的serialVersionUID,和序列化時生成的serialVersionUID進行比對,兩個serialVersionUID相同則反序列化成功,否則就會拋異常。

而當顯示的設置serialVersionUID後,Java在序列化和反序列化對象時,生成的serialVersionUID都為我們設定的serialVersionUID,這樣就保證了反序列化的成功。

transient

序列化對象時如果希望哪個屬性不被序列化,則用transient關鍵字修飾即可

@Data
public class User implements Serializable {

    private transient String name;

    private String age;
}

可以看到欄位name的值沒有被保存到磁碟中,一旦變數被transient修飾,變數將不再是對象持久化的一部分,該變數內容在序列化後無法獲得訪問。

Java序列化前的結果: User(name=fufu, age=18)
Java反序列化的結果:User(name=null, age=18)

一個靜態變數不管是否被transient修飾,均不能被序列化。 因為static修飾的屬性是屬於類,而非對象。

總結

分享了一個很小的知識點,工作再忙也不要忘了溫故而知新哦


今天就說這麼多,如果本文對您有一點幫助,希望能得到您一個點贊

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

-Advertisement-
Play Games
更多相關文章
  • 正常搭建配置中心,網上教程多,這裡不討論,只記坑也是為了後來者少花時間在這裡,由於是當時研究了好久才寫的文章,所以只能提供問題的原因,當然會給出印證的思路,閑話不多說進入正題! 版本spring boot2.1.6spring boot admin 2.1.6spring cloud GreenWi ...
  • 一、解決IDEA中的中文亂碼 1、首先設置idea編輯器的編碼: File-Setting設置如下 idea顯示編碼:windows預設用gbk所以idea顯示預設為gbk編碼,在 Help--Edit custom vm options 添加-Dfile.encoding=UTF-8 (其實大部分 ...
  • 開發環境: Windows操作系統開發工具: Eclipse+Jdk+Tomcat+MySql資料庫運行效果圖 源碼及原文鏈接:https://javadao.xyz/forum.php?mod=viewthread&tid=47 ...
  • 關註公眾號:CoderBuff,回覆“redis”獲取《Redis5.x入門教程》完整版PDF。 《Redis5.x入門教程》目錄 "第一章 · 準備工作" "第二章 · 數據類型" "第三章 · ​命令" "第四章 ​· 配置" 第五章 · Java客戶端(上) 第六章 · 事務 第七章 · 分佈 ...
  • 1.設置django的mysql驅動為pymysql 因為django預設的是使用MySqlDb連接mysql資料庫,但是由於該模塊不支持python3.4以上版本,所以使用pymysql模塊 在項目的預設文件夾(也就是settings.py等文件的那個文件夾)的__init__.py文件中增加如下 ...
  • 更新多個對象 例如說我們現在想要將Apress Publisher的名稱由原來的”Apress”更改為”Apress Publishing”。若使用save()方法,如: ? 1 2 3 >>> p = Publisher.objects.get(name='Apress') >>> p.name  ...
  • 最近的Django項目中有複製記錄的需求。資料庫里有一張名為Party的表,記錄用戶創建的party,現在要讓用戶能夠複製一個新的party。本身非常簡單的一個功能,但運行的時候出錯了。我以為是複製過程出錯,所以測試了一下Django中複製記錄可能遇到的情況(後來發現是其他代碼的邏輯出了問題),結果 ...
  • 前言: 公司的產品是一款基於社交的內容聊天軟體,需要集成語音通訊功能,在寫iOS原生項目時,用到的就是Agora SDK,現在寫React Native也直接採用了Agora的庫。 集成iOS、Android的步驟: 請參考鏈接:https://github.com/syanbo/react-nat ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...