成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

你知道為什么序列化要寫serialVersionUID嗎?

開發(fā) 前端
Externalizable?接口想必大家很少用到,它是Serializable?接口的子類,用戶要實現(xiàn)的writeExternal()和readExternal() 方法,用來決定如何序列化和反序列化。

?前言

java中的序列化可能大家像我一樣都停留在實現(xiàn)Serializable?接口上,對于它里面的一些核心機制沒有深入了解過。直到最近在項目中踩了一個坑,就是序列化對象添加一個字段以后,使用方系統(tǒng)報了反序列化失敗,原因是我們雙方的序列化對象沒有加上serialVersionUID,那你們知道下面幾個問題嗎:

  1. 序列化對象中的serialVersionUID 是干嘛用的?
  2. 如何修改默認(rèn)的序列化機制?
  3. 如何使用序列化的方式克隆對象?

對象序列化和反序列化機制

序列化: 將對象轉(zhuǎn)成二進制寫到輸出流的過程。

反序列化: 通過輸入流讀回二進制轉(zhuǎn)成對象的過程。

通過對象的序列化和反序列化機制可以實現(xiàn)對象在網(wǎng)絡(luò)之間傳輸。

在Java中,如果一個對象要想實現(xiàn)序列化,必須要實現(xiàn)下面兩個接口之一:

  • Serializable 接口
  • Externalizable 接口

這里我們先講解常用的Serializable 接口。

writeObject序列化過程栗子:

@Test
public void testSerializable() throws FileNotFoundException {
User user = new User("alvin", 19);
// 文件輸出流
FileOutputStream bout = new FileOutputStream("user.dat");
try (ObjectOutputStream out = new ObjectOutputStream(bout)) {
// 序列化
out.writeObject(user);
} catch (IOException e) {
e.printStackTrace();
}
}


@Data
@NoArgsConstructor
@AllArgsConstructor
public class User implements Serializable {

private String username;

private Integer age;
}

結(jié)果:

圖片

readObject反序列化栗子:

現(xiàn)在模擬另外一個系統(tǒng)需要反序列化user.dat

@Test
public void testDeSerializable() throws FileNotFoundException {
User user = null;
// 寫到內(nèi)存中,當(dāng)然也可以寫到文件中
FileInputStream fis = new FileInputStream("user.dat");
try (ObjectInputStream in = new ObjectInputStream(fis)) {
// 反序列化 readObject
user = (User) in.readObject();
} catch (IOException | ClassNotFoundException e) {
e.printStackTrace();
}

Assert.assertEquals("alvin", user.getUsername());
}

如果User類不實現(xiàn)Serializable接口, 那會怎么樣?

當(dāng)然是報錯了,如下圖:

圖片

小結(jié):

一個對象想要被序列化,那么它的類就要實現(xiàn)此接口或者它的子接口。

修改默認(rèn)的序列化機制

默認(rèn)的情況下,如果實現(xiàn)了Serializable接口的對象進行序列化的時候,默認(rèn)會將全部的數(shù)據(jù)域,也就是成員變量進行序列化輸出,那往往有時候并不需要這樣,有什么方法可以修改序列化機制呢?下面提供3種方式。

使用transient關(guān)鍵字

將成員變量標(biāo)記成transient,那么在序列化的過程中這些數(shù)據(jù)域會被跳過,如下圖所示:

圖片

這是一種最簡單的方式,但是不夠靈活。

自定義readObject、writeObject方法

序列化類中可以通過定義下面簽名的方法:

  • private void readObject(java.io.ObjectInputStream s) throws java.io.IOException, ClassNotFoundException
  • private void writeObject(java.io.ObjectOutputStream s) throws java.io.IOException

只要類中有這兩個簽名的方法,那么就不會調(diào)用默認(rèn)的序列化,取而代之調(diào)用這些方法。

本例我們舉個jdk中的例子,ArrayList就實現(xiàn)了這兩個方法,重寫了序列化機制。

圖片

主要原因ArrayList底層的數(shù)組通常會預(yù)留一些容量,等容量不足時再擴充容量,那么有些空間可能就沒有實際存儲元素,采用自定義方式實現(xiàn)序列化時,就可以保證只序列化實際存儲的那些元素,而不是整個數(shù)組,從而節(jié)省空間和時間。

實現(xiàn)Externalizable接口

Externalizable?接口想必大家很少用到,它是Serializable?接口的子類,用戶要實現(xiàn)的writeExternal()和readExternal() 方法,用來決定如何序列化和反序列化。

因為序列化和反序列化方法需要自己實現(xiàn),因此可以指定序列化哪些屬性,而transient在這里無效。

對Externalizable?對象反序列化時,會先調(diào)用類的無參構(gòu)造方法,這是有別于默認(rèn)反序列方式的。如果把類的不帶參數(shù)的構(gòu)造方法刪除,或者把該構(gòu)造方法的訪問權(quán)限設(shè)置為private?、默認(rèn)或protected?級別,會拋出java.io.InvalidException: no valid constructor?異常,因此Externalizable?對象必須有默認(rèn)構(gòu)造函數(shù),而且必需是public的。

舉例說明:

public class User2 implements Externalizable {

private String username;

private Integer age;

@Override
public void writeExternal(ObjectOutput out) throws IOException {
out.writeUTF(username);
out.writeInt(age);
}

@Override
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
username = in.readUTF();
age = in.readInt();
}
}

serialVersionUID的作用

這就回到概述中提到的項目中遇到的問題,現(xiàn)在簡要描述下:

A系統(tǒng)中的序列化對象User用的最新版本如下:

圖片

B系統(tǒng)中反序列化的對象,還是老的User版本如下:

圖片

這時候A系統(tǒng)生成的序列化文件,交給B系統(tǒng)反序列化時,出錯了, 如下圖:

圖片

原因:

類定義發(fā)生了變化,比如添加、刪除、修改類中的數(shù)據(jù)域后,它的唯一標(biāo)記符或者稱為SHA指紋、或者理解為serialVersionUID都會發(fā)生變化,這個值會保存在序列化二進制中,如果反序列化過程發(fā)現(xiàn)對不上,就會報錯,如上圖所示。

那么如何處理呢?

這時候,我們?nèi)绻X得這個序列化對象是可以兼容的,那么可以自定義一個serialVersionUID的靜態(tài)成員變量,它就不會自動生成,而是直接用這個值,如下圖:

圖片

使用序列化clone

clone大家都知道吧,在深拷貝的時候編碼還是很麻煩的,借用序列化機制可以實現(xiàn)深拷貝。做法很簡單,就是將對象序列化到輸出流中,然后讀回。

public class SerialCloneable implements Cloneable, Serializable {

@Override
public Object clone() throws CloneNotSupportedException {
try {
// 保存到字節(jié)數(shù)組流
ByteArrayOutputStream bout = new ByteArrayOutputStream();
try(ObjectOutputStream out = new ObjectOutputStream(bout)) {
out.writeObject(this);
}
// 讀取
try(InputStream bin = new ByteArrayInputStream(bout.toByteArray())) {
ObjectInputStream in = new ObjectInputStream(bin);
return in.readObject();
}
} catch (IOException | ClassNotFoundException e) {
CloneNotSupportedException e2 = new CloneNotSupportedException();
e2.initCause(e);
throw e2;
}
}
}

圖片

注意一點,這種方式性能不高,通常比顯示構(gòu)建、復(fù)制數(shù)據(jù)要慢不少。

總結(jié)

本文講解了序列化的一些核心機制,不再簡簡單單的停留在序列化就是實現(xiàn)Serializable接口了,希望能幫助到大家。

責(zé)任編輯:武曉燕 來源: JAVA旭陽
相關(guān)推薦

2024-04-03 09:23:31

ES索引分析器

2018-10-28 15:40:23

Python編程語言

2023-06-26 08:20:02

openapi格式注解

2023-09-08 08:35:42

層疊樣式表CSS

2014-07-15 11:05:30

黑莓

2020-12-30 05:33:30

Java網(wǎng)絡(luò)數(shù)據(jù)

2011-06-01 15:18:43

Serializabl

2020-12-24 18:46:11

Java序列化編程語言

2024-09-03 08:17:59

2024-12-02 12:34:06

2020-11-17 08:30:06

LinuxSwapping 設(shè)計

2023-11-02 10:22:29

gRPC后端通信

2024-10-09 08:19:35

2019-12-09 10:29:04

Go語言開發(fā)互聯(lián)網(wǎng)

2022-08-06 08:41:18

序列化反序列化Hessian

2025-05-08 08:30:00

Redis自定義序列化數(shù)據(jù)庫

2025-04-30 10:49:11

Java序列化反序列化

2025-03-05 10:49:32

2020-04-16 11:19:55

深度學(xué)習(xí)神經(jīng)網(wǎng)絡(luò)網(wǎng)絡(luò)層

2025-06-05 01:11:00

點贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 欧美精品中文字幕久久二区 | 国产精品一级 | 久久精品二区 | 青青草一区二区 | 91精品国产乱码麻豆白嫩 | 91精品久久久久 | 亚洲欧美日韩在线不卡 | 日韩精品一区二区三区中文字幕 | 日韩一区二区av | 久久国产精品99久久久大便 | 久久这里有精品 | 亚洲精品欧美一区二区三区 | 国产精品中文字幕在线播放 | 日日操日日舔 | 亚洲欧美成人 | 亚洲精品成人网 | 丁香五月网久久综合 | 久久久久久天堂 | 一区二区三区亚洲 | 久久久久久久国产精品视频 | 久久九精品 | 久久亚洲一区二区 | 精品入口麻豆88视频 | 精品久久香蕉国产线看观看亚洲 | 日韩电影一区 | 三级视频网站 | 免费看a | 91精品国产91久久综合桃花 | 久久精品一区二区三区四区 | 日韩在线视频免费观看 | 国产成人99久久亚洲综合精品 | 性视频网 | 日韩欧美国产精品一区二区 | 日韩综合网 | 精品一区电影 | 一级a性色生活片久久毛片波多野 | av免费网站在线 | 成人在线观看网站 | 日韩一区精品 | 亚洲一区二区电影网 | 中文字幕免费在线 |