日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程语言 > java >内容正文

java

[转载] Java序列化的几种方式以及序列化的作用

發(fā)布時(shí)間:2025/3/11 java 21 豆豆
生活随笔 收集整理的這篇文章主要介紹了 [转载] Java序列化的几种方式以及序列化的作用 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

參考鏈接: Java中帶有繼承的對象序列化

文章轉(zhuǎn)載自:? 本文著重講解一下Java序列化的相關(guān)內(nèi)容。?

如果對Java序列化感興趣的同學(xué)可以研究一下。?

一.Java序列化的作用?

有的時(shí)候我們想要把一個(gè)Java對象變成字節(jié)流的形式傳出去,有的時(shí)候我們想要從一個(gè)字節(jié)流中恢復(fù)一個(gè)Java對象。例如,有的時(shí)候我們想要?

把一個(gè)Java對象寫入到硬盤或者傳輸?shù)骄W(wǎng)路上面的其它計(jì)算機(jī),這時(shí)我們就需要自己去通過java把相應(yīng)的對象寫成轉(zhuǎn)換成字節(jié)流。對于這種通用?

的操作,我們?yōu)槭裁床皇褂媒y(tǒng)一的格式呢?沒錯,這里就出現(xiàn)了java的序列化的概念。在Java的OutputStream類下面的子類ObjectOutput-?

Stream類就有對應(yīng)的WriteObject(Object object) 其中要求對應(yīng)的object實(shí)現(xiàn)了java的序列化的接口。?

為了更好的理解java序列化的應(yīng)用,我舉兩個(gè)自己在開發(fā)項(xiàng)目中遇到的例子:?

1)在使用tomcat開發(fā)JavaEE相關(guān)項(xiàng)目的時(shí)候,我們關(guān)閉tomcat后,相應(yīng)的session中的對象就存儲在了硬盤上,如果我們想要在tomcat重啟的?

時(shí)候能夠從tomcat上面讀取對應(yīng)session中的內(nèi)容,那么保存在session中的內(nèi)容就必須實(shí)現(xiàn)相關(guān)的序列化操作。?

2)如果我們使用的java對象要在分布式中使用或者在rmi遠(yuǎn)程調(diào)用的網(wǎng)絡(luò)中使用的話,那么相關(guān)的對象必須實(shí)現(xiàn)java序列化接口。?

親愛的小伙伴,大概你已經(jīng)了解了java序列化相關(guān)的作用,接下來們來看看如何實(shí)現(xiàn)java的序列化吧。~?

二.實(shí)現(xiàn)java對象的序列化和反序列化。?

? ? ? ?Java對象的序列化有兩種方式。

?

? ? ? ?a.是相應(yīng)的對象實(shí)現(xiàn)了序列化接口Serializable,這個(gè)使用的比較多,對于序列化接口Serializable接口是一個(gè)空的接口,它的主要作用就是

?

? ? ? ? ?標(biāo)識這個(gè)對象時(shí)可序列化的,jre對象在傳輸對象的時(shí)候會進(jìn)行相關(guān)的封裝。這里就不做過多的介紹了。

?

? ? ? ? ?下面是一個(gè)實(shí)現(xiàn)序列化接口的Java序列化的例子:非常簡單

?

package com.shop.domain;

?

import java.util.Date;

?

?

public class Article implements java.io.Serializable {

? ? private static final long serialVersionUID = 1L;

? ? private Integer id;?

? ? private String title;? //文章標(biāo)題

? ? private String content;? // 文章內(nèi)容

? ? private String faceIcon;//表情圖標(biāo)

? ? private Date postTime; //文章發(fā)表的時(shí)間

? ? private String ipAddr;? //用戶的ip

?

? ? private User author;? //回復(fù)的用戶

?

? ? public Integer getId() {

? ? ? ? return id;

? ? }

? ? public void setId(Integer id) {

? ? ? ? this.id = id;

? ? }

? ? public String getTitle() {

? ? ? ? return title;

? ? }

? ? public void setTitle(String title) {

? ? ? ? this.title = title;

? ? }

? ? public String getContent() {

? ? ? ? return content;

? ? }

? ? public void setContent(String content) {

? ? ? ? this.content = content;

? ? }

? ? public String getFaceIcon() {

? ? ? ? return faceIcon;

? ? }

? ? public void setFaceIcon(String faceIcon) {

? ? ? ? this.faceIcon = faceIcon;

? ? }

? ? public Date getPostTime() {

? ? ? ? return postTime;

? ? }

? ? public void setPostTime(Date postTime) {

? ? ? ? this.postTime = postTime;

? ? }

? ? public User getAuthor() {

? ? ? ? return author;

? ? }

? ? public void setAuthor(User author) {

? ? ? ? this.author = author;

? ? }

? ? public String getIpAddr() {

? ? ? ? return ipAddr;

? ? }

? ? public void setIpAddr(String ipAddr) {

? ? ? ? this.ipAddr = ipAddr;

? ? }

?

?

}?

  b.實(shí)現(xiàn)序列化的第二種方式為實(shí)現(xiàn)接口Externalizable,Externlizable的部分源代碼如下:?

* @see java.io.ObjectInput

?* @see java.io.Serializable

?* @since? ?JDK1.1

?*/

public interface Externalizable extends java.io.Serializable {

? ? /**

? ? ?* The object implements the writeExternal method to save its contents

? ? ?* by calling the methods of DataOutput for its primitive values or?

?首先,我們在序列化對象的時(shí)候,由于這個(gè)類實(shí)現(xiàn)了Externalizable 接口,在writeExternal()方法里定義了哪些屬性可以序列化,

?

哪些不可以序列化,所以,對象在經(jīng)過這里就把規(guī)定能被序列化的序列化保存文件,不能序列化的不處理,然后在反序列的時(shí)候自動調(diào)?

用readExternal()方法,根據(jù)序列順序挨個(gè)讀取進(jìn)行反序列,并自動封裝成對象返回,然后在測試類接收,就完成了反序列。?

?所以說Exterinable的是Serializable的一個(gè)擴(kuò)展。

?

?為了更好的理解相關(guān)內(nèi)容,請看下面的例子:

?

package com.xiaohao.test;

?

import java.io.Externalizable;

import java.io.FileInputStream;

import java.io.FileNotFoundException;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.ObjectInput;

import java.io.ObjectInputStream;

import java.io.ObjectOutput;

import java.io.ObjectOutputStream;

import java.text.SimpleDateFormat;

import java.util.Date;

?

?

/**

?* 測試實(shí)體類

?* @author 小浩

?* @創(chuàng)建日期 2015-3-12

?*/

class Person implements Externalizable{

? ? ? ? private static final long serialVersionUID = 1L;<br>? ? String userName;

? ? String password;

? ? String age;

?

?

? ? public Person(String userName, String password, String age) {

? ? ? ? super();

? ? ? ? this.userName = userName;

? ? ? ? this.password = password;

? ? ? ? this.age = age;

? ? }

?

?

? ? public Person() {

? ? ? ? super();

? ? }

?

?

? ? public String getAge() {

? ? ? ? return age;

? ? }

? ? public void setAge(String age) {

? ? ? ? this.age = age;

? ? }

? ? public String getUserName() {

? ? ? ? return userName;

? ? }

? ? public void setUserName(String userName) {

? ? ? ? this.userName = userName;

? ? }

? ? public String getPassword() {

? ? ? ? return password;

? ? }

? ? public void setPassword(String password) {

? ? ? ? this.password = password;

? ? }

?

? ? /**

? ? ?* 序列化操作的擴(kuò)展類

? ? ?*/

? ? @Override

? ? public void writeExternal(ObjectOutput out) throws IOException {

? ? ? ? //增加一個(gè)新的對象

? ? ? ? Date date=new Date();

? ? ? ? out.writeObject(userName);

? ? ? ? out.writeObject(password);

? ? ? ? out.writeObject(date);

? ? }

?

? ? /**

? ? ?* 反序列化的擴(kuò)展類

? ? ?*/

? ? @Override

? ? public void readExternal(ObjectInput in) throws IOException,

? ? ? ? ? ? ClassNotFoundException {

? ? ? ? //注意這里的接受順序是有限制的哦,否則的話會出錯的

? ? ? ? // 例如上面先write的是A對象的話,那么下面先接受的也一定是A對象...

? ? ? ? userName=(String) in.readObject();

? ? ? ? password=(String) in.readObject();

? ? ? ? SimpleDateFormat sdf=new SimpleDateFormat("yyyy-MM-dd");

? ? ? ? Date date=(Date)in.readObject();? ? ? ?

? ? ? ? System.out.println("反序列化后的日期為:"+sdf.format(date));

?

? ? }

? ? @Override

? ? public String toString() {

? ? ? ? //注意這里的年齡是不會被序列化的,所以在反序列化的時(shí)候是讀取不到數(shù)據(jù)的

? ? ? ? return "用戶名:"+userName+"密 碼:"+password+"年齡:"+age;

? ? }

}

?

?

/**

?* 序列化和反序列化的相關(guān)操作類

?* @author 小浩

?* @創(chuàng)建日期 2015-3-12

?*/

class Operate{

? ? /**

? ? ?* 序列化方法

? ? ?* @throws IOException

? ? ?* @throws FileNotFoundException

? ? ?*/

? ? public void serializable(Person person) throws FileNotFoundException, IOException{

? ? ? ? ObjectOutputStream outputStream=new ObjectOutputStream(new FileOutputStream("a.txt"));

? ? ? ? outputStream.writeObject(person);? ? ??

? ? }

?

? ? /**

? ? ?* 反序列化的方法

? ? ?* @throws IOException

? ? ?* @throws FileNotFoundException

? ? ?* @throws ClassNotFoundException

? ? ?*/

? ? public Person deSerializable() throws FileNotFoundException, IOException, ClassNotFoundException{

? ? ? ? ObjectInputStream ois=new ObjectInputStream(new FileInputStream("a.txt"));

? ? ? ? return (Person) ois.readObject();

? ? }

?

?

?

}

/**

?* 測試實(shí)體主類

?* @author 小浩

?* @創(chuàng)建日期 2015-3-12

?*/

public class Test{

? ? public static void main(String[] args) throws FileNotFoundException, IOException, ClassNotFoundException {

? ? ? ?Operate operate=new Operate();

? ? ? ?Person person=new Person("小浩","123456","20");

? ? ? ?System.out.println("為序列化之前的相關(guān)數(shù)據(jù)如下:\n"+person.toString());

? ? ? ?operate.serializable(person);

? ? ? ?Person newPerson=operate.deSerializable();

? ? ? ?System.out.println("-------------------------------------------------------");

? ? ? ?System.out.println("序列化之后的相關(guān)數(shù)據(jù)如下:\n"+newPerson.toString());

? ? }

?

?

}?

首先,我們在序列化UserInfo對象的時(shí)候,由于這個(gè)類實(shí)現(xiàn)了Externalizable 接口,在writeExternal()方法里定義了哪些屬性可?

以序列化,哪些不可以序列化,所以,對象在經(jīng)過這里就把規(guī)定能被序列化的序列化保存文件,不能序列化的不處理,然后在反序列?

的時(shí)候自動調(diào)用readExternal()方法,根據(jù)序列順序挨個(gè)讀取進(jìn)行反序列,并自動封裝成對象返回,然后在測試類接收,就完成了反?

序列。?

?***對于實(shí)現(xiàn)Java的序列化接口需要注意一下幾點(diǎn):

?

? ? ? ?1.java中的序列化時(shí)transient變量(這個(gè)關(guān)鍵字的作用就是告知JAVA我不可以被序列化)和靜態(tài)變量不會被序列

?

? ? ? ? ? 化(下面是一個(gè)測試的例子)

?

import java.io.*;

?

class Student1 implements Serializable {

? ? private static final long serialVersionUID = 1L;

? ? private String name;

? ? private transient String password;

? ? private static int count = 0;

?

? ? public Student1(String name, String password) {

? ? ? ? System.out.println("調(diào)用Student的帶參的構(gòu)造方法");

? ? ? ? this.name = name;

? ? ? ? this.password = password;

? ? ? ? count++;

? ? }

?

? ? public String toString() {

? ? ? ? return "人數(shù): " + count + " 姓名: " + name + " 密碼: " + password;

? ? }

}

?

public class ObjectSerTest1 {

? ? public static void main(String args[]) {

? ? ? ? try {

? ? ? ? ? ? FileOutputStream fos = new FileOutputStream("test.obj");

? ? ? ? ? ? ObjectOutputStream oos = new ObjectOutputStream(fos);

? ? ? ? ? ? Student1 s1 = new Student1("張三", "12345");

? ? ? ? ? ? Student1 s2 = new Student1("王五", "54321");

? ? ? ? ? ? oos.writeObject(s1);

? ? ? ? ? ? oos.writeObject(s2);

? ? ? ? ? ? oos.close();

? ? ? ? ? ? FileInputStream fis = new FileInputStream("test.obj");

? ? ? ? ? ? ObjectInputStream ois = new ObjectInputStream(fis);

? ? ? ? ? ? Student1 s3 = (Student1) ois.readObject();

? ? ? ? ? ? Student1 s4 = (Student1) ois.readObject();

? ? ? ? ? ? System.out.println(s3);

? ? ? ? ? ? System.out.println(s4);

? ? ? ? ? ? ois.close();

? ? ? ? } catch (IOException e) {

? ? ? ? ? ? e.printStackTrace();

? ? ? ? } catch (ClassNotFoundException e1) {

? ? ? ? ? ? e1.printStackTrace();

? ? ? ? }

? ? }

}?

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.ObjectInputStream;

import java.io.ObjectOutputStream;

?

public class Test{

?

?

?

? ? public static void main(String args[]){

?

? ? ? ? try {

?

? ? ? ? ? ? FileInputStream fis = new FileInputStream("test.obj");

? ? ? ? ? ? ObjectInputStream ois = new ObjectInputStream(fis);

?

? ? ? ? ? ? Student1 s3 = (Student1) ois.readObject();

? ? ? ? ? ? Student1 s4 = (Student1) ois.readObject();

?

? ? ? ? ? ? System.out.println(s3);

? ? ? ? ? ? System.out.println(s4);

?

? ? ? ? ? ? ois.close();

? ? ? ? } catch (IOException e) {

? ? ? ? ? ? e.printStackTrace();

? ? ? ? } catch (ClassNotFoundException e1) {

? ? ? ? ? ? e1.printStackTrace();

? ? ? ? }

? ? }

?

?

?

}?

? ? ? ? ? ? 2.也是最應(yīng)該注意的,如果你先序列化對象A后序列化B,那么在反序列化的時(shí)候一定記著JAVA規(guī)定先讀到的對象

?

? ? ? ? ? ? ? ?是先被序列化的對象,不要先接收對象B,那樣會報(bào)錯.尤其在使用上面的Externalizable的時(shí)候一定要注意讀取

?

? ? ? ? ? ? ? ?的先后順序。

?

? ? ? ? ? ? 3.實(shí)現(xiàn)序列化接口的對象并不強(qiáng)制聲明唯一的serialVersionUID,是否聲明serialVersionUID對于對象序列化的向

?

? ? ? ? ? ? ? 上向下的兼容性有很大的影響。我們來做個(gè)測試:

?

思路一? 把User中的serialVersionUID去掉,序列化保存。反序列化的時(shí)候,增加或減少個(gè)字段,看是否成功。? Java代碼?

public class User implements Serializable{

?

private String name;

?

?private int age;

?

private long phone;

?

private List<UserVo> friends;

?

...<br>}?

保存到文件中:?

Java代碼

ByteArrayOutputStream bos = new ByteArrayOutputStream();

ObjectOutputStream os = new ObjectOutputStream(bos);

os.writeObject(src);

os.flush();

os.close();

byte[] b = bos.toByteArray();

bos.close();

FileOutputStream fos = new FileOutputStream(dataFile);

fos.write(b);

fos.close();?

增加或者減少字段后,從文件中讀出來,反序列化:?

Java代碼

ByteArrayOutputStream bos = new ByteArrayOutputStream();

ObjectOutputStream os = new ObjectOutputStream(bos);

os.writeObject(src);

os.flush();

os.close();

byte[] b = bos.toByteArray();

bos.close();

FileOutputStream fos = new FileOutputStream(dataFile);

fos.write(b);

fos.close();?

結(jié)果:拋出異常信息? Java代碼?

Exception in thread "main" java.io.InvalidClassException: serialize.obj.UserVo; local class incompatible: stream classdesc serialVersionUID = 3305402508581390189, local class serialVersionUID = 7174371419787432394 at java.io.ObjectStreamClass.initNonProxy(ObjectStreamClass.java:560)

at java.io.ObjectInputStream.readNonProxyDesc(ObjectInputStream.java:1582)

at java.io.ObjectInputStream.readClassDesc(ObjectInputStream.java:1495)

at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1731)

at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1328)

at java.io.ObjectInputStream.readObject(ObjectInputStream.java:350)

at serialize.obj.ObjectSerialize.read(ObjectSerialize.java:74)

at serialize.obj.ObjectSerialize.main(ObjectSerialize.java:27)

  ?

思路二? eclipse指定生成一個(gè)serialVersionUID,序列化保存,修改字段后反序列化? 略去代碼? 結(jié)果:反序列化成功? 結(jié)論? 如果沒有明確指定serialVersionUID,序列化的時(shí)候會根據(jù)字段和特定的算法生成一個(gè)serialVersionUID,當(dāng)屬性有變化時(shí)這個(gè)id發(fā)生了變化,所以反序列化的時(shí)候? 就會失敗。拋出“本地classd的唯一id和流中class的唯一id不匹配”。? jdk文檔關(guān)于serialVersionUID的描述:? 寫道? 如果可序列化類未顯式聲明 serialVersionUID,則序列化運(yùn)行時(shí)將基于該類的各個(gè)方面計(jì)算該類的默認(rèn) serialVersionUID 值,如“Java(TM) 對象序列化規(guī)范”中所述。不過,強(qiáng)烈建議 所有可序列化類都顯式聲明 serialVersionUID 值,原因是計(jì)算默認(rèn)的 serialVersionUID 對類的詳細(xì)信息具有較高的敏感性,根據(jù)編譯器實(shí)現(xiàn)的不同可能千差萬別,這樣在反序列化過程中可能會導(dǎo)致意外的 InvalidClassException。因此,為保證 serialVersionUID 值跨不同 java 編譯器實(shí)現(xiàn)的一致性,序列化類必須聲明一個(gè)明確的 serialVersionUID 值。還強(qiáng)烈建議使用 private 修飾符顯示聲明 serialVersionUID(如果可能),原因是這種聲明僅應(yīng)用于直接聲明類 – serialVersionUID 字段作為繼承成員沒有用處。數(shù)組類不能聲明一個(gè)明確的 serialVersionUID,因此它們總是具有默認(rèn)的計(jì)算值,但是數(shù)組類沒有匹配 serialVersionUID 值的要求。?

三.實(shí)現(xiàn)序列化的其它方式 (這是一個(gè)擴(kuò)展內(nèi)容,感興趣的可以擴(kuò)展一下)??

? ?1)是把對象包裝成JSON字符串傳輸。

?

? ? ?這里采用JSON格式同時(shí)使用采用Google的gson-2.2.2.jar 進(jìn)行轉(zhuǎn)義

?

? 2)采用谷歌的ProtoBuf

?

? ? ?隨著Google工具protoBuf的開源,protobuf也是個(gè)不錯的選擇。對JSON,Object Serialize(Java的序列化和反序列化),

?

? ? ?ProtoBuf 做個(gè)對比。

?

? ? ?定義一個(gè)通用的待傳輸?shù)膶ο骍serVo:

?

public class User

<span style="white-space: pre;">private static final long serialVersionUID = -5726374138698742258L;</span>

{ private String name;

? private int age;

? private long phone;

? private List<User> friends;

?...set和get方法

?}

 ?

初始化User的實(shí)例src:?

User user1 = new UserVo();

user1 .setName("user1 ");

?user1 .setAge(30);

?user1 .setPhone(13789126278L);

?UserVo f1 = new UserVo();

?f1.setName("tmac");

?f1.setAge(32);

?f1.setPhone(123L);

?User user2 = new User();

?user2 .setName("user2 ");

?user2 .setAge(29);

?user2 .setPhone(123L); <br> List<User> friends = new ArrayList<User>();

?friends.add(user1 );

?friends.add(user2 );

?user1 .setFriends(friends);?

1.首先使用JOSN來實(shí)現(xiàn)序列化。?

Gson gson = new Gson();<br>String json = gson.toJson(src);?

得到的字符串:?

{"name":"user1 ","age":30,"phone":123,"friends":[{"name":"user1 ","age":32,"phone":123},{"name":"user2 ","age":29,"phone":123}]}?

字節(jié)數(shù)為153? Json的優(yōu)點(diǎn):明文結(jié)構(gòu)一目了然,可以跨語言,屬性的增加減少對解析端影響較小。缺點(diǎn):字節(jié)數(shù)過多,依賴于不同的第三方類庫。?

Object Serialize(Java的序列化和反序列化)? UserVo實(shí)現(xiàn)Serializalbe接口,提供唯一的版本號:? 序列化方法:?

ByteArrayOutputStream bos = new ByteArrayOutputStream();

ObjectOutputStream os = new ObjectOutputStream(bos);

os.writeObject(src);

os.flush();

os.close();

byte[] b = bos.toByteArray();

bos.close();?

字節(jié)數(shù)是238?

反序列化:?

ObjectInputStream ois = new ObjectInputStream(fis);

vo = (UserVo) ois.readObject();

ois.close();

fis.close();?

Object Serializalbe 優(yōu)點(diǎn):java原生支持,不需要提供第三方的類庫,使用比較簡單。? 缺點(diǎn):無法跨語言,字節(jié)數(shù)占用比較大,某些情況下對于對象屬性的變化比較敏感。? 對象在進(jìn)行序列化和反序列化的時(shí)候,必須實(shí)現(xiàn)Serializable接口,但并不強(qiáng)制聲明唯一的serialVersionUID? 是否聲明serialVersionUID對于對象序列化的向上向下的兼容性有很大的影響。?

Google ProtoBuf? protocol buffers 是google內(nèi)部得一種傳輸協(xié)議,目前項(xiàng)目已經(jīng)開源(http://code.google.com/p/protobuf/)。? 它定義了一種緊湊得可擴(kuò)展得二進(jìn)制協(xié)議格式,適合網(wǎng)絡(luò)傳輸,并且針對多個(gè)語言有不同得版本可供選擇。? 以protobuf-2.5.0rc1為例,準(zhǔn)備工作:? 下載源碼,解壓,編譯,安裝? Shell代碼? tar zxvf protobuf-2.5.0rc1.tar.gz ./configure ./make ./make install? 測試:? Shell代碼? MacBook-Air:~ ming$ protoc –version libprotoc 2.5.0? 安裝成功!? 進(jìn)入源碼得java目錄,用mvn工具編譯生成所需得jar包,protobuf-java-2.5.0rc1.jar?

1、編寫.proto文件,命名UserVo.proto?

?

package serialize;

option java_package = "serialize";

option java_outer_classname="UserVoProtos";

message User{

optional string name = 1;

optional int32 age = 2;

optional int64 phone = 3;

repeated serialize.UserVo friends = 4;

?

}?

2、在命令行利用protoc 工具生成builder類? Shell代碼? protoc -IPATH=.proto文件所在得目錄 –java_out=java文件的輸出路徑 .proto的名稱? 得到UserProtos類?

3、編寫序列化代碼?

UserVoProtos.User.Builder builder = UserVoProtos.User.newBuilder();

builder.setName("Yaoming"); builder.setAge(30);

builder.setPhone(13789878978L);

UserVoProtos.UserVo.Builder builder1 = UserVoProtos.UserVo.newBuilder();

builder1.setName("tmac"); builder1.setAge(32); builder1.setPhone(138999898989L);

UserVoProtos.UserVo.Builder builder2 = UserVoProtos.UserVo.newBuilder();

builder2.setName("liuwei"); builder2.setAge(29); builder2.setPhone(138999899989L);

builder.addFriends(builder1);

builder.addFriends(builder2);

UserVoProtos.UserVo vo = builder.build();? byte[] v = vo.toByteArray();?

字節(jié)數(shù)53? 反序列化?

UserVoProtos.UserVo uvo = UserVoProtos.UserVo.parseFrom(dstb);

System.out.println(uvo.getFriends(0).getName());?

結(jié)果:tmac,反序列化成功? google protobuf 優(yōu)點(diǎn):字節(jié)數(shù)很小,適合網(wǎng)絡(luò)傳輸節(jié)省io,跨語言 。? 缺點(diǎn):需要依賴于工具生成代碼。?

工作機(jī)制? proto文件是對數(shù)據(jù)的一個(gè)描述,包括字段名稱,類型,字節(jié)中的位置。protoc工具讀取proto文件生成對應(yīng)builder代碼的類庫。protoc xxxxx –java_out=xxxxxx 生成java類庫。builder類根據(jù)自己的算法把數(shù)據(jù)序列化成字節(jié)流,或者把字節(jié)流根據(jù)反射的原理反序列化成對象。官方的示例:https://developers.google.com/protocol-buffers/docs/javatutorial。? proto文件中的字段類型和java中的對應(yīng)關(guān)系:? 詳見:https://developers.google.com/protocol-buffers/docs/proto? .proto Type java Type c++ Type? double double double? float float float? int32 int int32? int64 long int64? uint32 int uint32? unint64 long uint64? sint32 int int32? sint64 long int64? fixed32 int uint32? fixed64 long uint64? sfixed32 int int32? sfixed64 long int64? bool boolean bool? string String string? bytes byte string? 字段屬性的描述:? 寫道? required: a well-formed message must have exactly one of this field. optional: a well-formed message can have zero or one of this field (but not more than one). repeated: this field can be repeated any number of times (including zero) in a well-formed message. The order of the repeated values will be preserved.? protobuf 在序列化和反序列化的時(shí)候,是依賴于.proto文件生成的builder類完成,字段的變化如果不表現(xiàn)在.proto文件中就不會影響反序列化,比較適合字段變化的情況。? 做個(gè)測試:把UserVo序列化到文件中:?

UserProtos.User vo = builder.build();

byte[] v = vo.toByteArray();

FileOutputStream fos = new FileOutputStream(dataFile);

fos.write(vo.toByteArray());

fos.close();

?

?

為User增加字段,對應(yīng)的.proto文件:

?

Text代碼

?

package serialize;

option java_package = "serialize";

option java_outer_classname="UserVoProtos";

message User{

optional string name = 1;

optional int32 age = 2;

optional int64 phone = 3;

repeated serialize.UserVo friends = 4;

optional string address = 5; }

?

?

?

從文件中反序列化回來:

?

Java代碼

FileInputStream fis = new FileInputStream(dataFile);

byte[] dstb = new byte[fis.available()];

for(int i=0;i<dstb.length;i++){ dstb[i] = (byte)fis.read(); }

fis.close(); UserProtos.User uvo = UserProtos.User.parseFrom(dstb);

System.out.println(uvo.getFriends(0).getName());?

從文件中反序列化回來:?

Java代碼?

FileInputStream fis = new FileInputStream(dataFile);

byte[] dstb = new byte[fis.available()];

for(int i=0;i<dstb.length;i++){ dstb[i] = (byte)fis.read(); }

fis.close(); UserProtos.User uvo = UserProtos.User.parseFrom(dstb);

System.out.println(uvo.getFriends(0).getName());?

成功得到結(jié)果。?

三種方式對比傳輸同樣的數(shù)據(jù),google protobuf只有53個(gè)字節(jié)是最少的。結(jié)論:?

方式 優(yōu)點(diǎn) 缺點(diǎn)? JSON? 跨語言、格式清晰一目了然? 字節(jié)數(shù)比較大,需要第三方類庫? Object Serialize java原生方法不依賴外部類庫 字節(jié)數(shù)比較大,不能跨語言? Google protobuf? 跨語言、字節(jié)數(shù)比較少? 編寫.proto配置用protoc工具生成對應(yīng)的代碼

總結(jié)

以上是生活随笔為你收集整理的[转载] Java序列化的几种方式以及序列化的作用的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。