序列化和反序列化理解

其實(shí)作為一名程序員,必須要知道序列化反序列化 的概念,因?yàn)樗窃诠ぷ髦蟹浅3R姷母拍睢?/p>

1. 什么是序列化?反序列化?

Java 序列化就是指將對(duì)象轉(zhuǎn)換為字節(jié)序列的過(guò)程,而反序列化則是只將字節(jié)序列轉(zhuǎn)換成目標(biāo)對(duì)象的過(guò)程。

seriallization 序列化 : 將對(duì)象轉(zhuǎn)化為便于傳輸?shù)母袷剑?常見的序列化格式:二進(jìn)制格式,字節(jié)數(shù)組,json字符串,xml字符串。
deseriallization 反序列化:將序列化的數(shù)據(jù)恢復(fù)為對(duì)象的過(guò)程。

舉個(gè)例子:
比如:現(xiàn)在我們都會(huì)在淘寶上買桌子,桌子這種很不規(guī)則不東西,該怎么從一個(gè)城市運(yùn)輸?shù)搅硪粋€(gè)城市,這時(shí)候一般都會(huì)把它拆掉成板子,再裝到箱子里面,就可以快遞寄出去了,這個(gè)過(guò)程就類似我們的序列化的過(guò)程(把數(shù)據(jù)轉(zhuǎn)化為可以存儲(chǔ)或者傳輸?shù)男问剑.?dāng)買家收到貨后,就需要自己把這些板子組裝成桌子的樣子,這個(gè)過(guò)程就像反序列 的過(guò)程(轉(zhuǎn)化成當(dāng)初的數(shù)據(jù)對(duì)象)。

2、為什么要序列化?

我們都知道,在進(jìn)行瀏覽器訪問(wèn)的時(shí)候,我們看到的文本、圖片、音頻、視頻等都是通過(guò)二進(jìn)制序列進(jìn)行傳輸?shù)?,那么如果我們需要將Java對(duì)象進(jìn)行傳輸?shù)臅r(shí)候,是不是也應(yīng)該先將對(duì)象進(jìn)行序列化?答案是肯定的,我們需要先將Java對(duì)象進(jìn)行序列化,然后通過(guò)網(wǎng)絡(luò),IO進(jìn)行傳輸,當(dāng)?shù)竭_(dá)目的地之后,再進(jìn)行反序列化獲取到我們想要的對(duì)象,最后完成通信。

3. 例子

//創(chuàng)建student對(duì)象
        Student student = new Student(1,"1234","小明");

        //序列化
        byte[] bytes = toBytes(student);
        System.out.println(bytes.length);

        //反序列化
        Student student0 = (Student) toObj(bytes);
        System.out.println(student0);

輸出結(jié)果:

111
Student{id=1, code='1234', name='小明'}

不過(guò)我個(gè)人覺(jué)得,這樣序列化為二進(jìn)制格式,對(duì)于可視化的體驗(yàn)就很差了,如果這樣存到庫(kù)里,那這個(gè)具體是個(gè)什么鬼,就一眼看不出來(lái)了。

4. 如何實(shí)現(xiàn)序列化

jackson類庫(kù)的JSON操作方法:ObjectMapper

Java下常見的Json類庫(kù)有Gson、JSON-lib和Jackson等,Jackson相對(duì)來(lái)說(shuō)比較高效,在項(xiàng)目中主要使用Jackson進(jìn)行JSON和Java對(duì)象轉(zhuǎn)換,下面給出一些Jackson的JSON操作方法。

* ObjectMapper是JSON操作的核心,Jackson的所有JSON操作都是在ObjectMapper中實(shí)現(xiàn)。 
* ObjectMapper有多個(gè)JSON序列化的方法,可以把JSON字符串保存File、OutputStream等不同的介質(zhì)中。 
* writeValue(File arg0, Object arg1)把a(bǔ)rg1轉(zhuǎn)成json序列,并保存到arg0文件中。 
* writeValue(OutputStream arg0, Object arg1)把a(bǔ)rg1轉(zhuǎn)成json序列,并保存到arg0輸出流中。 
* writeValueAsBytes(Object arg0)把a(bǔ)rg0轉(zhuǎn)成json序列,并把結(jié)果輸出成字節(jié)數(shù)組。 
* writeValueAsString(Object arg0)把a(bǔ)rg0轉(zhuǎn)成json序列,并把結(jié)果輸出成字符串。 

所以,一般會(huì)將某個(gè)對(duì)象或者數(shù)組變成json字符串,存放在數(shù)據(jù)庫(kù)中,舉個(gè)栗子:

  • java 對(duì)象轉(zhuǎn)JSON(JSON序列化)
//JSON序列化和反序列化使用的User類 
public class User {  
    private String name;  
    private Integer age;  
    private Date birthday;  
    private String email;
}
import java.io.IOException;  
import java.text.ParseException;  
import java.text.SimpleDateFormat;  
  
import com.fasterxml.jackson.databind.ObjectMapper;  
  
public class JacksonDemo {
        User user = new User();  
        user.setName("小民");   
        user.setEmail("xiaomin@sina.com");  
        user.setAge(20);  
          
        SimpleDateFormat dateformat = new SimpleDateFormat("yyyy-MM-dd");  
        user.setBirthday(dateformat.parse("1996-10-01"));  
        ObjectMapper mapper = new ObjectMapper();  
          
        //User類轉(zhuǎn)JSON  : 序列化為JSON字符串序列
        //輸出結(jié)果:{"name":"小民","age":20,"birthday":844099200000,"email":"xiaomin@sina.com"}  
        String json = mapper.writeValueAsString(user);  
        System.out.println(json);  
          
        //Java集合轉(zhuǎn)JSON  
        //輸出結(jié)果:[{"name":"小民","age":20,"birthday":844099200000,"email":"xiaomin@sina.com"}]  
        List<User> users = new ArrayList<User>();  
        users.add(user);  
        String jsonlist = mapper.writeValueAsString(users);  
        System.out.println(jsonlist);  
       }

    }  
  • JSON轉(zhuǎn)Java類[JSON反序列化]
import java.io.IOException;  
import java.text.ParseException;  
import com.fasterxml.jackson.databind.ObjectMapper;  
  
public class JacksonDemo {  
    public static void main(String[] args) throws ParseException, IOException {  
        String json = "{\"name\":\"小民\",\"age\":20,\"birthday\":844099200000,\"email\":\"xiaomin@sina.com\"}";  
          
        /** 
         * ObjectMapper支持從byte[]、File、InputStream、字符串等數(shù)據(jù)的JSON反序列化。 
         */  
        ObjectMapper mapper = new ObjectMapper();  
        User user = mapper.readValue(json, User.class);  
        System.out.println(user);  
    }  
} 
  • 字符串?dāng)?shù)組序列化 和反序列化 :
import com.fasterxml.jackson.databind.ObjectMapper;

ObjectMapper objectMapper = new ObjectMapper();
String[] originStrArray = {"a", "b","b"};
// json 序列化為json格式的字符串序列
String  str = objectMapper.writeValueAsString(originStrArray);
System.out.println(str);

// 反序列化為字符串?dāng)?shù)組
String[] originStrArray  = objectMapper.readValue(strArray, String[].class);
System.out.println(originStrArray.length);

打印結(jié)果:



這樣我們就可以直接這樣作為json字符串存庫(kù)

從庫(kù)里拿出來(lái),直接進(jìn)行反轉(zhuǎn)就可以有搖身一變就是字符串?dāng)?shù)組。

還有一種方式:

import com.google.gson.Gson;

//  數(shù)組序列化為json字符串序列
Gson gson = new Gson();
String[] strings = new String[]{"123", "456", "abc"};
String json = gson.toJson(strings);
System.out.println("" + json);

// 反序列化 成字符串?dāng)?shù)組
String[] strings2 = gson.fromJson(json, String[].class);
for (String s : strings2)
    System.out.println("" + s);

5. 如何保證序列化和反序列化后的對(duì)象一致?(如有異議望指正)

對(duì)于這個(gè)問(wèn)題我在查閱了一些資料之后,發(fā)現(xiàn)并不能保證序列化和反序列化之后的對(duì)象是一致的,因?yàn)槲覀冊(cè)诜葱蛄谢倪^(guò)程中,是先創(chuàng)建一個(gè)對(duì)象,然后再通過(guò)對(duì)對(duì)象進(jìn)行賦值來(lái)完成對(duì)象的反序列化,這樣問(wèn)題就來(lái)了,在創(chuàng)建了一個(gè)新的對(duì)象之后,對(duì)象引用和原本的對(duì)象并不是指向同一個(gè)目標(biāo)。因此我們只能保證他們的數(shù)據(jù)和版本一致,并不能保證對(duì)象一致。

6. JSON注解

Jackson提供了一系列注解,方便對(duì)JSON序列化和反序列化進(jìn)行控制,下面介紹一些常用的注解。

@JsonIgnore 此注解用于屬性上,作用是進(jìn)行JSON操作時(shí)忽略該屬性。

@JsonFormat 此注解用于屬性上,作用是把Date類型直接轉(zhuǎn)化為想要的格式,如@JsonFormat(pattern = "yyyy-MM-dd HH-mm-ss")。

@JsonProperty 此注解用于屬性上,作用是把該屬性的名稱序列化為另外一個(gè)名稱,如把trueName屬性序列化為name,@JsonProperty("name")。

import java.util.Date;  
import com.fasterxml.jackson.annotation.*;  
  
public class User {  
    private String name;  
      
    //不JSON序列化年齡屬性  
    @JsonIgnore   
    private Integer age;  
      
    //格式化日期屬性  
    @JsonFormat(pattern = "yyyy年MM月dd日")  
    private Date birthday;  
      
    //序列化email屬性為mail  
    @JsonProperty("mail")  
    private String email;  
}  
  
import java.io.IOException;  
import java.text.ParseException;  
import java.text.SimpleDateFormat;  
  
import com.fasterxml.jackson.databind.ObjectMapper;  
  
public class JacksonDemo {  
  
    public static void main(String[] args) throws ParseException, IOException {  
        User user = new User();  
        user.setName("小民");   
        user.setEmail("xiaomin@sina.com");  
        user.setAge(20);  
          
        SimpleDateFormat dateformat = new SimpleDateFormat("yyyy-MM-dd");  
        user.setBirthday(dateformat.parse("1996-10-01"));         
          
        ObjectMapper mapper = new ObjectMapper();  
        String json = mapper.writeValueAsString(user);  
        System.out.println(json);  
        //輸出結(jié)果:{"name":"小民","birthday":"1996年09月30日","mail":"xiaomin@sina.com"}  
    }  
}  
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,697評(píng)論 19 139
  • JAVA序列化機(jī)制的深入研究 對(duì)象序列化的最主要的用處就是在傳遞,和保存對(duì)象(object)的時(shí)候,保證對(duì)象的完整...
    時(shí)待吾閱讀 11,224評(píng)論 0 24
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 47,290評(píng)論 6 342
  • EventBus EventBus是針一款對(duì)Android的發(fā)布/訂閱事件總線。它可以讓我們很輕松的實(shí)現(xiàn)在Andr...
    安卓小生閱讀 407評(píng)論 2 1
  • 青春漸漸走遠(yuǎn),而曾經(jīng)卻不曾忘記。讀書的大部分時(shí)間里,我們亦步亦趨的做一些正兒八經(jīng)的事,而少部分時(shí)間里,我們傳...
    魚塘小八閱讀 609評(píng)論 0 0

友情鏈接更多精彩內(nèi)容