欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文帶你徹底理解Java序列化和反序列化

 更新時(shí)間:2020年09月03日 09:07:07   作者:mySoul  
這篇文章主要介紹了Java序列化和反序列化的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)Java,感興趣的朋友可以了解下

Java序列化是什么?

Java序列化是指把Java對(duì)象轉(zhuǎn)換為字節(jié)序列的過(guò)程,Java反序列化是指把字節(jié)序列恢復(fù)為Java對(duì)象的過(guò)程。

反序列化: 客戶端重文件,或者網(wǎng)絡(luò)中獲取到文件以后,在內(nèi)存中重構(gòu)對(duì)象。

序列化: 對(duì)象序列化的最重要的作用是傳遞和保存對(duì)象的時(shí)候,保證對(duì)象的完整性和可傳遞性。方便字節(jié)可以在網(wǎng)絡(luò)上傳輸以及保存在本地文件。

為什么需要序列化和反序列化

實(shí)現(xiàn)分布式

核心在于RMI,可以利用對(duì)象序列化運(yùn)行遠(yuǎn)程主機(jī)上的服務(wù),實(shí)現(xiàn)運(yùn)行的時(shí)候,就像在本地上運(yùn)行Java對(duì)象一樣。

實(shí)現(xiàn)遞歸保存對(duì)象

進(jìn)行序列化的時(shí)候,單單并不是保存一個(gè)對(duì)象,而是遞歸的保存一整個(gè)對(duì)象序列,即遞歸保存,通過(guò)反序列化,可以遞歸的得到一整個(gè)對(duì)象序列。

序列信息可以永久保存

用于序列化的信息,可以永久保存為文件,或者保存在數(shù)據(jù)庫(kù)中,在使用的時(shí)候,再次隨時(shí)恢復(fù)到內(nèi)存中,實(shí)現(xiàn)內(nèi)存中的類(lèi)的信息可以永久的保存。

數(shù)據(jù)格式統(tǒng)一

比照Linux的一切皆文件的思想,同時(shí)Java也是這樣的思想,讓數(shù)據(jù)格式盡可能的統(tǒng)一,讓對(duì)象,文件,數(shù)據(jù),等等許許多多不同的格式,都讓其統(tǒng)一,以及保存。實(shí)現(xiàn)數(shù)據(jù)可以完整的傳輸和保存。然后進(jìn)行反序列化還原,即,對(duì)象還是對(duì)象,文件還是文件。

實(shí)現(xiàn)Java序列化和反序列化

要進(jìn)行反序列化需要實(shí)現(xiàn)一個(gè)接口。即 Serializabei接口。

代碼如下

需要轉(zhuǎn)化的類(lèi)

package common.lang;

import java.io.Serializable;

import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.lang3.builder.ToStringStyle;

public class User1 implements Serializable{

 private String name;
 private int age;

 public String getName() {
  return name;
 }
 public void setName(String name) {
  this.name = name;
 }
 public int getAge() {
  return age;
 }
 public void setAge(int age) {
  this.age = age;
 }

 @Override
 public String toString() {
  return new ToStringBuilder(this, ToStringStyle.MULTI_LINE_STYLE)
         .append("name", name)
         .append("age", age)
         .toString();
 }
}

進(jìn)行序列化,以及反序列化

package common.lang;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

public class SerializableDemo1 {

 public static void main(String[] args) throws Exception, IOException {
  //初始化對(duì)象
  User1 user = new User1();
  user.setName("yaomy");
  user.setAge(23);
  System.out.println(user);
  //序列化對(duì)象到文件中
  ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("template"));
  oos.writeObject(user);
  oos.close();
  //反序列化
  File file = new File("template");
  ObjectInputStream ois = new ObjectInputStream(new FileInputStream(file));
  User1 newUser = (User1)ois.readObject();
  System.out.println(newUser.toString());
 }
}

另一個(gè)序列化接口 Externalizable

繼續(xù)實(shí)現(xiàn)Externalizable接口

package common.lang;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;

import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.lang3.builder.ToStringStyle;

public class User1 implements Externalizable{

 private String name;
 private int age;

 public String getName() {
  return name;
 }
 public void setName(String name) {
  this.name = name;
 }
 public int getAge() {
  return age;
 }
 public void setAge(int age) {
  this.age = age;
 }

 @Override
 public String toString() {
  return new ToStringBuilder(this, ToStringStyle.MULTI_LINE_STYLE)
         .append("name", name)
         .append("age", age)
         .toString();
 }
 @Override
 public void writeExternal(ObjectOutput out) throws IOException {
  // TODO Auto-generated method stub

 }
 @Override
 public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
  // TODO Auto-generated method stub

 }
}

進(jìn)行序列化以及反序列化

package common.lang;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

public class SerializableDemo1 {

 public static void main(String[] args) throws Exception, IOException {
  //初始化對(duì)象
  User1 user = new User1();
  user.setName("yaomy");
  user.setAge(23);
  System.out.println(user);
  //序列化對(duì)象到文件中
  ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("template"));
  oos.writeObject(user);
  oos.close();
  //反序列化
  File file = new File("template");
  ObjectInputStream ois = new ObjectInputStream(new FileInputStream(file));
  User1 newUser = (User1)ois.readObject();
  System.out.println(newUser.toString());
  ois.close();
 }
}

查看輸出的結(jié)構(gòu)

common.lang.User1@6ef64f64[
name=yaomy
age=23
]
common.lang.User1@184c9860[
name=<null>
age=0
]

根據(jù)輸出的結(jié)果可以看到,對(duì)User1進(jìn)行序列化然后再反序列化之后對(duì)象的屬性都恢復(fù)成了默認(rèn)值,即,之前那個(gè)對(duì)象的狀態(tài)沒(méi)有被持久保存下來(lái),這就是Externalization和Serialization接口的區(qū)別,其中前者接口會(huì)被恢復(fù)成為默認(rèn)值,后者接口不會(huì)恢復(fù)默認(rèn)值。

如果需要恢復(fù),這里就需要重寫(xiě)兩個(gè)抽象方法,分別是writeExternal與readExternal兩個(gè)抽象方法。

package common.lang;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;

import org.apache.commons.lang3.builder.ToStringBuilder;
import org.apache.commons.lang3.builder.ToStringStyle;

public class User1 implements Externalizable{

 private String name;
 private int age;

 public String getName() {
  return name;
 }
 public void setName(String name) {
  this.name = name;
 }
 public int getAge() {
  return age;
 }
 public void setAge(int age) {
  this.age = age;
 }

 @Override
 public String toString() {
  return new ToStringBuilder(this, ToStringStyle.MULTI_LINE_STYLE)
         .append("name", name)
         .append("age", age)
         .toString();
 }
 @Override
 public void writeExternal(ObjectOutput out) throws IOException {
  out.writeObject(name);
  out.writeInt(age);

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

 }
}
package common.lang;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;

public class SerializableDemo1 {

 public static void main(String[] args) throws Exception, IOException {
  //初始化對(duì)象
  User1 user = new User1();
  user.setName("yaomy");
  user.setAge(23);
  System.out.println(user);
  //序列化對(duì)象到文件中
  ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("template"));
  oos.writeObject(user);
  oos.close();
  //反序列化
  File file = new File("template");
  ObjectInputStream ois = new ObjectInputStream(new FileInputStream(file));
  User1 newUser = (User1)ois.readObject();
  System.out.println(newUser.toString());
  ois.close();
 }
}

輸出的結(jié)果

common.lang.User1@6cd66725[
name=yaomy
age=23
]
common.lang.User1@19160e64[
name=yaomy
age=23
]

靜態(tài)變量的序列化

實(shí)例

public class Test implements Serializable {

 private static final long serialVersionUID = 1L;

 public static int staticVar = 5;

 public static void main(String[] args) {
  try {
   //初始時(shí)staticVar為5
   ObjectOutputStream out = new ObjectOutputStream(
     new FileOutputStream("result.obj"));
   out.writeObject(new Test());
   out.close();

   //序列化后修改為10
   Test.staticVar = 10;

   ObjectInputStream oin = new ObjectInputStream(new FileInputStream(
     "result.obj"));
   Test t = (Test) oin.readObject();
   oin.close();

   //再讀取,通過(guò)t.staticVar打印新的值
   System.out.println(t.staticVar);

  } catch (FileNotFoundException e) {
   e.printStackTrace();
  } catch (IOException e) {
   e.printStackTrace();
  } catch (ClassNotFoundException e) {
   e.printStackTrace();
  }
 }
}

代碼闡述一下過(guò)程,在main方法中,對(duì)象序列化以后,修改靜態(tài)變量的數(shù)值,再把序列化后的對(duì)象讀取出來(lái),此時(shí)輸出的值為10.

理解如下: 打印的staticVar是從讀取對(duì)象里獲得的,打印10的原因是因?yàn)樾蛄谢瘯r(shí),不保存靜態(tài)變量,只保存內(nèi)存中的狀態(tài)。此時(shí)修改靜態(tài)變量的值,修改的是類(lèi)中的值,輸出的也是類(lèi)中的值,和內(nèi)存無(wú)關(guān)。

Transient關(guān)鍵字

Transient關(guān)鍵字,加上以后,可以阻止該變量被序列化到文件中,反序列化以后,變量的值設(shè)定為初始值。

以上就是一文帶你徹底理解Java序列化和反序列化的詳細(xì)內(nèi)容,更多關(guān)于Java序列化和反序列化的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java設(shè)計(jì)模式之外觀模式示例詳解

    Java設(shè)計(jì)模式之外觀模式示例詳解

    外觀模式為多個(gè)復(fù)雜的子系統(tǒng),提供了一個(gè)一致的界面,使得調(diào)用端只和這個(gè)接口發(fā)生調(diào)用,而無(wú)須關(guān)系這個(gè)子系統(tǒng)內(nèi)部的細(xì)節(jié)。本文將通過(guò)示例詳細(xì)為大家講解一下外觀模式,需要的可以參考一下
    2022-03-03
  • mybatis-plus?執(zhí)行insert(),實(shí)體的id自動(dòng)更新問(wèn)題

    mybatis-plus?執(zhí)行insert(),實(shí)體的id自動(dòng)更新問(wèn)題

    這篇文章主要介紹了mybatis-plus?執(zhí)行insert(),實(shí)體的id自動(dòng)更新問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • 一篇文章帶你了解Maven的坐標(biāo)概念以及依賴管理

    一篇文章帶你了解Maven的坐標(biāo)概念以及依賴管理

    這篇文章主要為大家介紹了Maven的坐標(biāo)概念以及依賴管理,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-01-01
  • 通過(guò)pipeline配置sonar自動(dòng)化實(shí)現(xiàn)過(guò)程解析

    通過(guò)pipeline配置sonar自動(dòng)化實(shí)現(xiàn)過(guò)程解析

    這篇文章主要介紹了通過(guò)pipeline配置sonar自動(dòng)化實(shí)現(xiàn)過(guò)程解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-11-11
  • ocp開(kāi)閉原則_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    ocp開(kāi)閉原則_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    這篇文章主要為大家詳細(xì)介紹了ocp開(kāi)閉原則的相關(guān)資料,ocp開(kāi)閉原則指導(dǎo)我們?nèi)绾谓⒁粋€(gè)穩(wěn)定的、靈活的系統(tǒng),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-08-08
  • 教你Java中的Lock鎖底層AQS到底是如何實(shí)現(xiàn)的

    教你Java中的Lock鎖底層AQS到底是如何實(shí)現(xiàn)的

    本文是基于ReentrantLock來(lái)講解,ReentrantLock加鎖只是對(duì)AQS的api的調(diào)用,底層的鎖的狀態(tài)(state)和其他線程等待(Node雙向鏈表)的過(guò)程其實(shí)是由AQS來(lái)維護(hù)的,對(duì)Java?Lock鎖AQS實(shí)現(xiàn)過(guò)程感興趣的朋友一起看看吧
    2022-05-05
  • 你知道將Bean交給Spring容器管理有幾種方式(推薦)

    你知道將Bean交給Spring容器管理有幾種方式(推薦)

    Spring核心是?IOC?和?AOP?,我們?cè)赟pring項(xiàng)目中,我們需要將Bean交給Spring容器,也就是IOC管理,這樣你才可以使用注解來(lái)進(jìn)行依賴注入,這篇文章主要介紹了你知道將Bean交給Spring容器管理有幾種方式,需要的朋友可以參考下
    2022-10-10
  • Spring中Bean命名的方式總結(jié)

    Spring中Bean命名的方式總結(jié)

    在?Spring?框架中,每個(gè)?bean?必須至少有一個(gè)唯一的名稱,這篇文章主要為大家詳細(xì)介紹了Spring中Bean命名的各種方式,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-12-12
  • spring中@ComponentScan自動(dòng)掃描并指定掃描規(guī)則

    spring中@ComponentScan自動(dòng)掃描并指定掃描規(guī)則

    本文主要介紹了spring中@ComponentScan自動(dòng)掃描并指定掃描規(guī)則,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • BufferedInputStream(緩沖輸入流)詳解_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    BufferedInputStream(緩沖輸入流)詳解_動(dòng)力節(jié)點(diǎn)Java學(xué)院整理

    這篇文章主要為大家詳細(xì)介紹了BufferedInputStream緩沖輸入流的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-05-05

最新評(píng)論