亚洲综合原千岁中文字幕_国产精品99久久久久久久vr_无码人妻aⅴ一区二区三区浪潮_成人h动漫精品一区二区三

主頁 > 知識庫 > J2SE中的序默認序列化

J2SE中的序默認序列化

熱門標簽:開封便宜外呼系統報價 地圖標注人員兼職 漯河辦理400電話 商丘百應電話機器人有沒有效果 電話機器人的特色和創新 怎樣把地圖標注出來 騰訊地圖標注商戶改名注冊入駐 淮南騰訊地圖標注 黃石智能營銷電銷機器人效果
要保存的也被保存了下來。一般情況下,我們僅僅需要保存邏輯數據就可以了。不需要保存的數據我們可以用關鍵字transient標出。

  以下是一個例子:

  import java.io.*;

  public class Serial implements Serializable {
  int company_id;
  String company_addr;

  transient boolean company_flag;
  }

  則company_flag字段將不會參與序列化與反序列化,但同時你也增加了為他初始值的責任。這也是序列化常常導致的問題之一。因為序列化相當于一個只接受數據流的public構造函數,這種對象構造方法是語言之外的。但他仍然是一種形式上的構造函數。如若你的類不能夠通過其他方面來保證初始化,則你需要額外的提供readObject方法,首先正常的反序列化,然后對transient標示的字段進行初始化。

  在不適合的時候,使用java默認的序列化行為可能會帶來速度上的影響,最糟糕的情況是,可能導致溢出。在某些數據結構的實現中,經常會充斥著各種的循環引用,而java的默認序列化行為,并不了解你的對象結構,其結果就是java試圖通過一種昂貴的“圖遍歷”來保存對象狀態。可想而知,不但慢而且可能溢出。這時候你就要提供自己的readObject,來代替默認的行為。

  兼容性問題

  兼容性歷來是復雜而麻煩的問題。

  不要兼容性:

  首先來看看如果我們的目的是不要兼容性,應該注意哪些。不要兼容性的場合很多,比如war3每當版本升級就不能夠讀取以前的replays。

  兼容也就是版本控制,java通過一個名為UID(stream unique identifier)來控制,這個UID是隱式的,它通過類名,方法名等諸多因素經過計算而得,理論上是一一映射的關系,也就是唯一的。如果UID不一樣的話,就無法實現反序列化了,并且將會得到InvalidClassException。

  當我們要人為的產生一個新的版本(實現并沒有改動),而拋棄以前的版本的話,可以通過顯式的聲名UID來實現:

  private static final long serialVersionUID=????;

  你可以編造一個版本號,但注意不要重復。這樣在反序列化的時候老版本將得到InvalidClassException,我們可以在老版本的地方捕捉這個異常,并提示用戶升級的新的版本。

  當改動不大時,保持兼容性(向下兼容性的一個特例):

  有時候你的類增加了一些無關緊要的非私有方法,而邏輯字段并不改變的時候,你當然希望老版本和新版本保持兼容性,方法同樣是通過顯式的聲名UID來實現。下面我們驗證一下。

  老版本:

  import java.io.*;

  public class Serial implements Serializable {

  int company_id;
  String company_addr;

  public Serial1(int company_id, String company_addr) {
  this.company_id = company_id;
  this.company_addr = company_addr;
  }

  public String toString() {
  return "DATA: "+company_id+" "+
  company_addr;
  }
  }

  新版本

  import java.io.*;

  public class Serial implements Serializable {

  int company_id;
  String company_addr;
  public Serial1(int company_id, String company_addr) {
  this.company_id = company_id;
  this.company_addr = company_addr;
  }

  public String toString() {
  return "DATA: "+company_id+" "+ company_addr;
  }
  public void todo(){}//無關緊要的方法
  }

  首先將老版本序列化,然后用新版本讀出,發生錯誤:

  java.io.InvalidClassException: Serial.Serial1; local class incompatible: stream classdesc serialVersionUID = 762508508425139227, local class serialVersionUID = 1187169935661445676

  接下來我們加入顯式的聲名UID:

  private static final long serialVersionUID=762508508425139227l;


  再次運行,順利地產生新對象

  DATA: 1001 com1

  如何保持向上兼容性:

  向上兼容性是指老的版本能夠讀取新的版本序列化的數據流。常常出現在我們的服務器的數據更新了,仍然希望老的客戶端能夠支持反序列化新的數據流,直到其更新到新的版本。可以說,這是半自動的事情。

  跟一般的講,因為在java中serialVersionUID是唯一控制著能否反序列化成功的標志,只要這個值不一樣,就無法反序列化成功。但只要這個值相同,無論如何都將反序列化,在這個過程中,對于向上兼容性,新數據流中的多余的內容將會被忽略;對于向下兼容性而言,舊的數據流中所包含的所有內容都將會被恢復,新版本的類中沒有涉及到的部分將保持默認值。利用這一特性,可以說,只要我們認為的保持serialVersionUID不變,向上兼容性是自動實現的。

  當然,一但我們將新版本中的老的內容拿掉,情況就不同了,即使UID保持不變,會引發異常。正是因為這一點,我們要牢記一個類一旦實現了序列化又要保持向上下兼容性,就不可以隨隨便便的修改了!!!

  測試也證明了這一點,有興趣的讀者可以自己試一試。
   如何保持向下兼容性:

  一如上文所指出的,你會想當然的認為只要保持serialVersionUID不變,向下兼容性是自動實現的。但實際上,向下兼容要復雜一些。這是因為,我們必須要對那些沒有初始化的字段負責。要保證它們能被使用。

  所以必須要利用

  private void readObject(java.io.ObjectInputStream in)
  throws IOException, ClassNotFoundException{
  in.defaultReadObject();//先反序列化對象
  if(ver=5552){
  //以前的版本5552
  …初始化其他字段
  }else if(ver=5550){
  //以前的版本5550
  …初始化其他字段
  }else{
  //太老的版本不支持
  throw new InvalidClassException();
  }
  }

  細心的讀者會注意到要保證in.defaultReadObject();能夠順利執行,就必須要求serialVersionUID保持一致,所以這里的ver不能夠利用serialVersionUID了。這里的ver是一個我們預先安插好的final long ver=xxxx;并且它不能夠被transient修飾。所以保持向下的兼容性至少有三點要求:

  1.serialVersionUID保持一致

  2.預先安插好我們自己的版本識別標志的final long ver=xxxx;

  3.保證初始化所有的域

  討論一下兼容性策略:

  到這里我們可以看到要保持向下的兼容性很麻煩。而且隨著版本數目的增加。維護會變得困難而繁瑣。討論什么樣的程序應該使用怎么樣的兼容性序列化策略已經超出本文的范疇,但是對于一個游戲的存盤功能,和對于一個字處理軟件的文檔的兼容性的要求肯定不同。對于rpg游戲的存盤功能,一般要求能夠保持向下兼容,這里如果使用java序列化的方法,則可根據以上分析的三點進行準備。對于這樣的情況使用對象序列化方法還是可以應付的。對于一個字處理軟件的文檔的兼容性要求頗高,一般情況下的策略都是要求良好的向下兼容性,和盡可能的向上兼容性。則一般不會使用對象序列化技術,一個精心設計的文檔結構,更能解決問題。

  數據一致性問題、約束問題

  要知道序列化是另一種形式上的“public構造函數”,但他僅僅構造起對象,而不作任何的檢查,這樣人很不舒服,所以必要的檢查是必須的,這利用了readObject()

  private void readObject(java.io.ObjectInputStream in)
  throws IOException, ClassNotFoundException{
  in.defaultReadObject();//先反序列化對象
  …進行檢查與初始化
  }

  出于結構化的考慮,通常使用一個名為initialize的函數,負責檢查與初始化,如果失敗拋出異常。要保持檢查與初始化是很容易被忘記的,這常常導致問題。另一個問題在于當父類沒有加入readObject()的時候,子類很容易忘記要調用對應的initialize函數。這仿佛回到了當初為什么要引入構造函數的問題,原因就是防止子類忘記調用初始化函數引發各種問題。所以,如果要保持數據一致性,一定要加入readObject()。

  安全問題

  安全性的話題超出了本文的范疇,但是你應該要知道,有可能一個攻擊者會對你的類準備一個惡意的數據流企圖生成一個錯誤的類。當你需要確保你的對象數據安全的話,你一般可以利用上面的方法來檢查,并初始化,但對于某些引用不好檢查。解決方法就是對重要的部件進行保護性拷貝。這里推薦一個好方法,它不用保護性拷貝個別的域,而是直接保護性拷貝整個對象。這就是:

  Object readResolve() throws ObjectStreamException;

  這個方法的用途就是,他會緊接著readObject()調用。它將會利用返回的對象代替原來反序列化的對象。也就是原來readObject()反序列化的對象將會被立即的丟棄。

  Object readResolve() throws ObjectStreamException{
  return new Serial2(this.xxx1,this.xxx2);// xxx1、xxx2是剛剛反序列化得來的,這是一種保護性拷貝
  }

  這樣的話雖然在時間上有所浪費,但是對于特別的重要而安全的類,可以使用這種方法。如果數據一致性問題、約束問題通過逐一檢查來解決很麻煩,也可以利用這種方法,但要考慮好成本,和注意下面的局限性。 利用readResolve()有一個明顯的缺點,就是當父類實現了readResolve(),子類將變得無叢下手。如果一個保護的或者是公有的父類的readResolve()存在,并且子類也沒有改寫它,將會使得子類反序列化的時候最終得到一個父類的對象,這既不是我們要得結果,也不容易發現這種錯誤。而讓子類重寫readResolve()無疑是一個負擔。也就是說對于要繼承的類而言,實現readResolve()來保護類不是一個好方法。我們只能利用第一種方法寫一個保護性的readObject()。

  所以我的建議是:一般情況下,只有對于final的類采用readResolve()來進行保護。

標簽:紅河 拉薩 馬鞍山 武威 亳州 大興安嶺 岳陽 鄭州

巨人網絡通訊聲明:本文標題《J2SE中的序默認序列化》,本文關鍵詞  J2SE,中的,序,默認,序列化,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《J2SE中的序默認序列化》相關的同類信息!
  • 本頁收集關于J2SE中的序默認序列化的相關信息資訊供網民參考!
  • 推薦文章
    国产精品123| 日韩中文字幕在线播放| 国产视频久久久| 日本在线不卡视频| 成人在激情在线视频| 精品视频免费在线| 国产极品白嫩美女在线观看看| 在线观看成人网| 国产精品自拍在线观看| 韩国三级视频网站| 午夜久久网| 国产网站在线| 欧美另类videosbestsex久久| 国产不卡高清| 成人免费观看男女羞羞视频| 尤物视频网站在线观看| 国产高清在线精品一区a| 亚洲www美色| 美女免费精品视频在线观看| 国产麻豆精品免费视频| 免费一级片在线观看| 国产欧美精品| 日日日夜夜操| 韩国毛片免费大片| 韩国三级视频网站| 成人免费高清视频| 日韩一级黄色大片| 成人免费观看网欧美片| 九九久久国产精品大片| 日本免费看视频| 欧美激情一区二区三区视频 | 国产综合成人观看在线| 成人av在线播放| 欧美a级片视频| 一级片片| 国产网站免费视频| 国产不卡高清| 欧美1区| 欧美夜夜骑 青草视频在线观看完整版 久久精品99无色码中文字幕 欧美日韩一区二区在线观看视频 欧美中文字幕在线视频 www.99精品 香蕉视频久久 | 成人高清视频免费观看| 成人高清视频在线观看| 一级片免费在线观看视频| 国产亚洲免费观看| 青青青草视频在线观看| 欧美a级v片不卡在线观看| 欧美a级片视频| 亚洲精品影院一区二区| 国产一区二区精品| 精品视频免费看| 国产成人精品影视| 九九九国产| 毛片高清| 成人免费观看的视频黄页| 美女免费精品视频在线观看| 亚欧视频在线| 久久国产一区二区| 精品久久久久久中文字幕一区| 韩国毛片免费大片| 国产不卡在线观看视频| 久久久久久久网| 一级女性全黄生活片免费| 91麻豆高清国产在线播放| 香蕉视频久久| 国产麻豆精品高清在线播放| 中文字幕97| 亚洲wwwwww| 精品视频在线观看一区二区| 国产一区二区精品尤物| 日本伦理黄色大片在线观看网站| 一级片免费在线观看视频| 亚洲天堂在线播放| 亚洲天堂在线播放| 国产一区二区高清视频| 九九九国产| 精品国产亚一区二区三区| 好男人天堂网 久久精品国产这里是免费 国产精品成人一区二区 男人天堂网2021 男人的天堂在线观看 丁香六月综合激情 | 国产麻豆精品免费密入口| 精品国产亚洲一区二区三区| 黄色免费三级| 韩国三级香港三级日本三级| 九九久久99综合一区二区| 欧美电影免费看大全| 欧美a免费| 91麻豆精品国产高清在线| 精品毛片视频| 天天做日日爱| 精品国产亚一区二区三区| 韩国毛片免费| 99久久网站| 日日夜夜婷婷| 国产麻豆精品| 99热精品在线| 91麻豆国产| 国产原创视频在线| 国产一区精品| 高清一级淫片a级中文字幕| 久久国产精品永久免费网站| 韩国毛片| 四虎久久精品国产| 日本免费乱理伦片在线观看2018| 国产伦精品一区三区视频| 久久国产精品只做精品| 成人a级高清视频在线观看| 久久久久久久男人的天堂| 午夜欧美成人久久久久久| 国产亚洲精品成人a在线| 色综合久久天天综线观看| 精品在线免费播放| 久久精品大片| 日韩一级黄色大片| 国产一区二区高清视频| 国产原创视频在线| 久久国产一久久高清| 黄视频网站免费| 欧美日本免费| 国产一区二区精品久久91| 国产成人啪精品| 欧美激情在线精品video| 你懂的国产精品| 日韩在线观看免费| 国产视频在线免费观看| 一级女性全黄久久生活片| 99色播| 日韩av成人| 国产亚洲精品成人a在线| 日韩在线观看免费| 国产一区二区精品| 夜夜操网| 欧美激情伊人| 欧美激情一区二区三区在线| 黄色免费三级| 精品视频在线观看免费| 国产成人啪精品| 99久久网站| 麻豆网站在线看| 黄视频网站免费观看| 免费国产在线视频| 国产高清在线精品一区二区| 日韩欧美一二三区| 日韩专区第一页| 成人高清免费| 免费一级片在线| 精品国产亚洲一区二区三区| 久久久久久久男人的天堂| 欧美爱爱网| 亚洲第一页色| 成人高清视频免费观看| 精品久久久久久中文字幕一区| 久久精品店| 国产成a人片在线观看视频| 亚欧成人乱码一区二区| 欧美a级片视频| 国产亚洲精品成人a在线| 亚欧视频在线| 欧美激情中文字幕一区二区| 尤物视频网站在线观看| 尤物视频网站在线观看| 亚洲www美色| 精品久久久久久综合网| 美女被草网站| 国产视频一区二区三区四区| 一a一级片| 亚洲精品中文一区不卡| 91麻豆国产级在线| 欧美激情中文字幕一区二区| 天天色色网| 99久久精品国产高清一区二区 | 欧美一级视频免费观看| 国产视频久久久| 欧美a级大片| 精品国产香蕉伊思人在线又爽又黄| 国产福利免费观看| 日韩一级黄色| 国产一区二区精品久久91| 欧美激情中文字幕一区二区| 精品国产亚洲一区二区三区| 精品国产亚洲人成在线| 国产一区二区精品久久| 免费国产在线观看| 久久99青青久久99久久| 国产一区二区精品| 黄视频网站免费观看| 久草免费资源| 精品毛片视频| 国产a免费观看| 国产亚洲免费观看| 久久久成人影院| 九九精品久久久久久久久| 欧美一区二区三区性| 91麻豆tv| 一级毛片视频免费| 日韩一级黄色| 成人影院一区二区三区| 欧美一级视| 天天做日日爱夜夜爽| 日韩一级黄色| 在线观看导航| 日本特黄特色aa大片免费| 欧美日本免费| 99久久精品国产片|