亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java中實(shí)體類為什么要實(shí)現(xiàn)Serializable序列化的作用

 更新時(shí)間:2020年11月06日 09:59:29   作者:二十同學(xué)  
這篇文章主要介紹了Java中實(shí)體類為什么要實(shí)現(xiàn)Serializable序列化的作用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

客戶端訪問(wèn)了某個(gè)能開啟會(huì)話功能的資源, web服務(wù)器就會(huì)創(chuàng)建一個(gè)與該客戶端對(duì)應(yīng)的HttpSession對(duì)象,每個(gè)HttpSession對(duì)象都要站用一定的內(nèi)存空間。如果在某一時(shí)間段內(nèi)訪問(wèn)站點(diǎn)的用戶很多,web服務(wù)器內(nèi)存中就會(huì)積累大量的HttpSession對(duì)象,消耗大量的服務(wù)器內(nèi)存,即使用戶已經(jīng)離開或者關(guān)閉了瀏覽器,web服務(wù)器仍要保留與之對(duì)應(yīng)的HttpSession對(duì)象,在他們超時(shí)之前,一直占用web服務(wù)器內(nèi)存資源。

        web服務(wù)器通常將那些暫時(shí)不活動(dòng)但未超時(shí)的HttpSession對(duì)象轉(zhuǎn)移到文件系統(tǒng)或數(shù)據(jù)庫(kù)中保存,服務(wù)器要使用他們時(shí)再將他們從文件系統(tǒng)或數(shù)據(jù)庫(kù)中裝載入內(nèi)存,這種技術(shù)稱為Session的持久化。

         將HttpSession對(duì)象保存到文件系統(tǒng)或數(shù)據(jù)庫(kù)中,需要采用序列化的方式將HttpSession對(duì)象中的每個(gè)屬性對(duì)象保存到文件系統(tǒng)或數(shù)據(jù)庫(kù)中;將HttpSession對(duì)象從文件系統(tǒng)或數(shù)據(jù)庫(kù)中裝載如內(nèi)存時(shí),需要采用反序列化的方式,恢復(fù)HttpSession對(duì)象中的每個(gè)屬性對(duì)象。所以存儲(chǔ)在HttpSession對(duì)象中的每個(gè)屬性對(duì)象必須實(shí)現(xiàn)Serializable接口。

serialVersionUID 的作用

serialVersionUID 用來(lái)表明類的不同版本間的兼容性

       Java的序列化機(jī)制是通過(guò)在運(yùn)行時(shí)判斷類的serialVersionUID來(lái)驗(yàn)證版本一致性的。在進(jìn)行反序列化時(shí),JVM會(huì)把傳來(lái)的字節(jié)流中的serialVersionUID與本地相應(yīng)實(shí)體(類)的serialVersionUID進(jìn)行比較,如果相同就認(rèn)為是一致的,可以進(jìn)行反序列化,否則就會(huì)出現(xiàn)序列化版本不一致的異常。

      當(dāng)實(shí)現(xiàn)java.io.Serializable接口的實(shí)體(類)沒(méi)有顯式地定義一個(gè)名為serialVersionUID,類型為long的變量時(shí),Java序列化機(jī)制會(huì)根據(jù)編譯的class自動(dòng)生成一個(gè)serialVersionUID作序列化版本比較用,這種情況下,只有同一次編譯生成的class才會(huì)生成相同的serialVersionUID 。

       如果我們不希望通過(guò)編譯來(lái)強(qiáng)制劃分軟件版本,即實(shí)現(xiàn)序列化接口的實(shí)體能夠兼容先前版本,未作更改的類,就需要顯式地定義一個(gè)名為serialVersionUID,類型為long的變量,不修改這個(gè)變量值的序列化實(shí)體都可以相互進(jìn)行串行化和反串行化。

      引起這個(gè)疑問(wèn),還是從Hibernate使用查詢緩存說(shuō)起;對(duì)象實(shí)例除了存在于內(nèi)存,二級(jí)緩存還會(huì)將對(duì)象寫進(jìn)硬盤在需要的時(shí)候再讀取出來(lái)使用,此時(shí)就必須提到一個(gè)概念:序列化。

      程序在運(yùn)行時(shí)實(shí)例化出對(duì)象,這些對(duì)象存在于內(nèi)存中,隨著程序運(yùn)行停止而消失,但如果我們想把某些對(duì)象(一般都是各不相同的屬性)保存下來(lái)或者傳輸給其他進(jìn)程,在程序終止運(yùn)行后這些對(duì)象仍然存在,可以在程序再次運(yùn)行時(shí)讀取這些對(duì)象的信息,或者在其他程序中利用這些保存下來(lái)的對(duì)象信息恢復(fù)成實(shí)例對(duì)象。這種情況下就要用到對(duì)象的序列化和反序列化。

 其實(shí)很早就知道的,在Java中常見(jiàn)的幾個(gè)類,如:Interger/String等,都實(shí)現(xiàn)了java.io.Serializable接口。這個(gè)序列化接口沒(méi)有任何方法和域,僅用于標(biāo)識(shí)序列化語(yǔ)意;實(shí)現(xiàn) Serializable 接口的類是可序列化的,沒(méi)有實(shí)現(xiàn)此接口的類將不能被序列化和反序列化。序列化類的所有子類本身都是可序列化的,不再需要顯式實(shí)現(xiàn) Serializable 接口。只有經(jīng)過(guò)序列化,才能兼容對(duì)象在磁盤文本以及在網(wǎng)絡(luò)中的傳輸,以及恢復(fù)對(duì)象的時(shí)候反序列化等操作。

問(wèn)題一:為何要實(shí)現(xiàn)序列化?

答:序列化就是對(duì)實(shí)例對(duì)象的狀態(tài)(State 對(duì)象屬性而不包括對(duì)象方法)進(jìn)行通用編碼(如格式化的字節(jié)碼)并保存,以保證對(duì)象的完整性和可傳遞性。

簡(jiǎn)而言之:序列化,就是為了在不同時(shí)間或不同平臺(tái)的JVM之間共享實(shí)例對(duì)象

// 經(jīng)常使用如下:
public static void main(String[] args) throws Exception {
  File file = new File("user.ser");
 
  ObjectOutputStream oout = new ObjectOutputStream(new FileOutputStream(file));
  User user = new User("zhang", 18, Gender.MALE);
  oout.writeObject(user);
  oout.close();
 
  ObjectInputStream oin = new ObjectInputStream(new FileInputStream(file));
  Object newUser = oin.readObject();
  oin.close();
  System.out.println(newUser);
}

  如沒(méi)有 實(shí)現(xiàn)Serializable接口,在序列化時(shí),使用ObjectOutputStream的write(object)方法將對(duì)象保存時(shí)將會(huì)出現(xiàn)異常。其實(shí) java.io.Serializable 只是一個(gè)沒(méi)有屬性和方法的空接口,但是問(wèn)題來(lái)了。。

問(wèn)題二:為何一定要實(shí)現(xiàn) Serializable 才能進(jìn)行序列化呢?

使用 ObjectOutputStream 來(lái)持久化對(duì)象, 對(duì)于此處拋出的異常,查看該類中實(shí)現(xiàn)如下:

private void writeObject0(Object obj, boolean unshared) throws IOException {
  // ...
      // 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());
        }
      }
  // ...
}

從此可知, 如果被寫對(duì)象類型是String、數(shù)組、Enum、Serializable,就可以進(jìn)行序列化,否則將拋出NotSerializableException。

最后提點(diǎn)注意:

1、在序列化對(duì)象時(shí),不僅會(huì)序列化當(dāng)前對(duì)象本身,還會(huì)對(duì)該對(duì)象引用的其它對(duì)象也進(jìn)行序列化,如此引用傳遞序列化。如果一個(gè)對(duì)象包含的成員變量是容器類等并深層引用,那么序列化過(guò)程開銷也較大。

2、當(dāng)字段被聲明為 transient 后,默認(rèn)序列化機(jī)制就會(huì)忽略該字段。(還有方法就是自定義writeObject方法,見(jiàn)下代碼示例)

3、在單例類中添加一個(gè)readResolve()方法(直接返回單例對(duì)象),以保證在序列化過(guò)程仍保持單例特性。

此外補(bǔ)充一下,

在路徑下jdk中還有另外一種形式的對(duì)象持久化,即:外部化(Externalization)。

public interface Externalizable extends java.io.Serializable {
 void writeExternal(ObjectOutput out) throws IOException;
 void readExternal(ObjectInput in) throws IOException, ClassNotFoundException;
}

外部化和序列化是實(shí)現(xiàn)同一目標(biāo)的兩種不同方法。

通過(guò) Serializable 接口對(duì)對(duì)象序列化的支持是jdk內(nèi)支持的 API ,但是java.io.Externalizable的所有實(shí)現(xiàn)者必須提供讀入和寫出的具體實(shí)現(xiàn),怎么實(shí)現(xiàn)完全由你自定義。序列化(Serializable )會(huì)自動(dòng)存儲(chǔ)所有必要的信息(如屬性以及屬性類型等),用以反序列化成原來(lái)一樣的實(shí)例,而外部化(Externalizable)則只保存被存儲(chǔ)實(shí)例中你需要的信息。

示例代碼如下:

public class User implements Externalizable {
  private String name;
  transient private Integer age; // 屏蔽字段
  private Gender gender;
 
  public User() {
    System.out.println("none constructor");
  }
 
  public User(String name, Integer age, Gender gender) {
    System.out.println("arg constructor");
    this.name = name;
    this.age = age;
    this.gender = gender;
  }
 
  // 實(shí)現(xiàn)讀寫
  private void writeObject(ObjectOutputStream out) throws IOException {
    out.defaultWriteObject();
    out.writeInt(age);
    // 屏蔽gender
  }
  private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
    in.defaultReadObject();
    age = in.readInt();
  }
 
  // 具體重寫
  @Override
  public void writeExternal(ObjectOutput out) throws IOException {
    out.writeObject(name);
    out.writeInt(age);
    // 屏蔽gender
  }
  @Override
  public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
    name = (String) in.readObject();
    age = in.readInt();
  } 
}

注意,用Externalizable進(jìn)行序列化,當(dāng)讀取對(duì)象時(shí),會(huì)調(diào)用被序列化類的無(wú)參構(gòu)造器創(chuàng)建一個(gè)新的對(duì)象,然后再將被保存對(duì)象的字段的值分別填充到新對(duì)象中。實(shí)現(xiàn)Externalizable接口的類必須要提供一個(gè)無(wú)參的構(gòu)造器,且訪問(wèn)權(quán)限為 public。

到此這篇關(guān)于Java中實(shí)體類為什么要實(shí)現(xiàn)Serializable序列化的作用的文章就介紹到這了,更多相關(guān)Java Serializable序列化內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Springcloud中Feign傳遞參數(shù)的過(guò)程解析

    Springcloud中Feign傳遞參數(shù)的過(guò)程解析

    這篇文章主要介紹了Springcloud中Feign傳遞參數(shù)的過(guò)程,單個(gè)參數(shù)的傳值有兩種方式,第一種使用@RequestParam/@PathVariable進(jìn)行傳值,傳遞多個(gè)參數(shù):多個(gè)參數(shù)的傳值可以使用多個(gè)@RequestParam來(lái)進(jìn)行傳參,需要的朋友可以參考下
    2023-09-09
  • springmvc實(shí)現(xiàn)跨服務(wù)器文件上傳功能

    springmvc實(shí)現(xiàn)跨服務(wù)器文件上傳功能

    這篇文章主要為大家詳細(xì)介紹了springmvc實(shí)現(xiàn)跨服務(wù)器文件上傳功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-08-08
  • Java關(guān)鍵字之instanceof詳解

    Java關(guān)鍵字之instanceof詳解

    instanceof是Java的一個(gè)二元操作符,和==,>,<是同一類東東。由于它是由字母組成的,所以也是Java的保留關(guān)鍵字。它的作用是測(cè)試它左邊的對(duì)象是否是它右邊的類的實(shí)例,返回boolean類型的數(shù)據(jù)
    2021-11-11
  • spring Cloud微服務(wù)跨域?qū)崿F(xiàn)步驟

    spring Cloud微服務(wù)跨域?qū)崿F(xiàn)步驟

    這篇文章主要介紹了spring Cloud微服務(wù)跨域?qū)崿F(xiàn)步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-11-11
  • Mybatis常用標(biāo)簽整理

    Mybatis常用標(biāo)簽整理

    日常開發(fā)中,MyBatis中標(biāo)簽有著舉足輕重的重要性,以下是一些MyBatis框架中常見(jiàn)的標(biāo)簽及案例,感興趣的朋友跟隨小編一起看看吧
    2007-02-02
  • Java類獲取Spring中bean的5種方式

    Java類獲取Spring中bean的5種方式

    這篇文章主要為大家詳細(xì)介紹了Java類獲取Spring中bean的5種方式,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-03-03
  • spring aop action中驗(yàn)證用戶登錄狀態(tài)的實(shí)例代碼

    spring aop action中驗(yàn)證用戶登錄狀態(tài)的實(shí)例代碼

    本篇文章主要介紹了spring aop action中驗(yàn)證用戶登錄狀態(tài)的實(shí)例代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-07-07
  • java按鈕控件數(shù)組實(shí)現(xiàn)計(jì)算器界面示例分享

    java按鈕控件數(shù)組實(shí)現(xiàn)計(jì)算器界面示例分享

    本文主要介紹了JAVA通過(guò)按鈕數(shù)組來(lái)管理界面中的所有按鈕控件,從而使用最少的代碼實(shí)現(xiàn)模擬的計(jì)算器界面
    2014-02-02
  • Java基于中介者模式實(shí)現(xiàn)多人聊天室功能示例

    Java基于中介者模式實(shí)現(xiàn)多人聊天室功能示例

    這篇文章主要介紹了Java基于中介者模式實(shí)現(xiàn)多人聊天室功能,詳細(xì)分析了中介者模式的概念、原理以及使用中介模式實(shí)現(xiàn)多人聊天的步驟、操作技巧與注意事項(xiàng),需要的朋友可以參考下
    2018-05-05
  • mybatis新增到數(shù)據(jù)庫(kù)后返回當(dāng)前ID問(wèn)題

    mybatis新增到數(shù)據(jù)庫(kù)后返回當(dāng)前ID問(wèn)題

    這篇文章主要介紹了mybatis新增到數(shù)據(jù)庫(kù)后返回當(dāng)前ID問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-08-08

最新評(píng)論