序列化和反序列化

目录

一、基本概念

1、定义:

2、作用:

3、数据传递:

4、序列化优点:

5、序列化算法需要做的事:

二、Java实现序列化和反序列化的过程

1、实现序列化的必备要求:

2、JDK中序列化和反序列化的API:

3、实现序列化和反序列化的三种实现:

4、Java代码示例:

5、图示:

三、扩展

1、如何保证序列化和反序列化后的对象一致:

2、JSON注解:


一、基本概念

1、定义:

(1)Java序列化是指Java对象转换为字节序列的过程;

(2)Java反序列化是指字节序列恢复为Java对象的过程。

2、作用:

(1)序列化最重要的作用:在传递和保存对象时,保证对象的完整性和可传递性。对象转换为有序字节流,以便在网络上传输或者保存在本地文件中;

(2)反序列化最重要的作用:根据字节流中保存的对象状态及描述信息,通过反序列化重建对象;

(3)总结:核心作用就是对象状态的保存和重建(整个过程核心点就是字节流中所保存的对象状态及描述信息)。

3、数据传递:

(1)在数据传输(网络传输)前,先通过序列化工具类将Java对象序列化为json/xml文件;

(2)在数据传输(网络传输)后,再将json/xml文件反序列化为对应语言的对象。

4、序列化优点:

(1)将对象转为字节流存储到硬盘上,当JVM停机的话,字节流还会在硬盘上默默等待,等待下一次JVM的启动,把序列化的对象,通过反序列化为原来的对象,并且序列化的二进制序列能够减少存储空间(永久性保存对象);

(2)序列化成字节流的对象可以进行网络传输(二进制形式),方便网络传输;

(3)通过序列化可以在进程间传递对象。

5、序列化算法需要做的事:

(1)将对象实例相关的类元数据输出;

(2)递归的输出类的超类描述直到不再有超类;

(3)类元数据输出完毕后,从最顶端的超类开始输出对象实例的实际数据值;

(4)从上至下递归输出实例的数据。

二、Java实现序列化和反序列化的过程

1、实现序列化的必备要求:

只有实现了Serializable或者Externalizable接口的类的对象才能被序列化为字节序列(不是则会抛出异常)。

2、JDK中序列化和反序列化的API:

(1)java.io.ObjectInputStream:对象输入流

该类的readObject()方法从输入流中读取字节序列,然后将字节序列反序列化为一个对象并返回

(2)java.io.ObjectOutputStream:对象输出流

该类的writeObject(Object obj)方法将将传入的obj对象进行序列化,把得到的字节序列写入到目标输出流中进行输出

3、实现序列化和反序列化的三种实现:

(1)若某个类仅仅实现了Serializable接口,则可以按照以下方式进行序列化和反序列化

        ①ObjectOutputStream采用默认的序列化方式,对该类对象的非transient的实例变量进行序列化;

        ②ObjcetInputStream采用默认的反序列化方式,对该类对象的非transient的实例变量进行反序列化。

(2)若某个类仅仅实现了Serializable接口,并且还定义了readObject(ObjectInputStream in)和writeObject(ObjectOutputSteam out),则采用以下方式进行序列化与反序列化

        ①ObjectOutputStream调用该类对象的writeObject(ObjectOutputStream out)的方法进行序列化;

        ②ObjectInputStream会调用该类对象的readObject(ObjectInputStream in)的方法进行反序列化。

(3)若某个类实现了Externalnalizable接口,且Student类必须实现readExternal(ObjectInput in)和writeExternal(ObjectOutput out)方法,则按照以下方式进行序列化与反序列化

        ① ObjectOutputStream调用该类对象的writeExternal(ObjectOutput out))的方法进行序列化;

        ② ObjectInputStream会调用该类对象的readExternal(ObjectInput in)的方法进行反序列化。

4、Java代码示例:


public class SerializableTest {
        public static void main(String[] args) throws IOException, ClassNotFoundException {
            //序列化
            FileOutputStream fos = new FileOutputStream("object.out");
            ObjectOutputStream oos = new ObjectOutputStream(fos);
            Student student1 = new Student("qingnianren", "1314gohard", "22");
            oos.writeObject(student1);
            oos.flush();
            oos.close();
            //反序列化
            FileInputStream fis = new FileInputStream("object.out");
            ObjectInputStream ois = new ObjectInputStream(fis);
            Student student2 = (Student) ois.readObject();
            System.out.println(student2.getUserName()+ " " +
                    student2.getPassword() + " " + student2.getYear());
    }
}

public class Student implements Serializable{                             
                                                                          
    private static final long serialVersionUID = -6060343040263809614L;   
                                                                          
    private String userName;                                              
    private String password;                                              
    private String year;                                                  
                                                                          
    public String getUserName() {                                         
        return userName;                                                  
    }                                                                     
                                                                          
    public String getPassword() {                                         
        return password;                                                  
    }                                                                     
                                                                          
    public void setUserName(String userName) {                            
        this.userName = userName;                                         
    }                                                                     
                                                                          
    public void setPassword(String password) {                            
        this.password = password;                                         
    }                                                                     
                                                                          
    public String getYear() {                                             
        return year;                                                      
    }                                                                     
                                                                          
    public void setYear(String year) {                                    
        this.year = year;                                                 
    }                                                                     
                                                                          
    public Student(String userName, String password, String year) {       
        this.userName = userName;                                         
        this.password = password;                                         
        this.year = year;                                                 
    }                                                                     
} 

5、图示:

(1)序列化图示

(2)反序列化图示 

三、扩展

1、如何保证序列化和反序列化后的对象一致:

不能保证序列化和反序列化之后的对象是一致的,因为我们在反序列化的过程中,是先创建一个对象,然后再通过对对象进行赋值来完成对象的反序列化,这样问题就来了,在创建了一个新的对象之后,对象引用和原本的对象并不是指向同一个目标。因此我们只能保证他们的数据和版本一致,并不能保证对象一致。

2、JSON注解:

Jackson提供了一系列注解,方便对JSON序列化和反序列化进行控制,下面介绍一些常用的注解:

(1)@JsonIgnore 此注解用于属性上,作用是进行JSON操作时忽略该属性;

(2)@JsonFormat 此注解用于属性上,作用是把Date类型直接转化为想要的格式,如@JsonFormat(pattern = "yyyy-MM-dd HH-mm-ss");

(3)@JsonProperty 此注解用于属性上,作用是把该属性的名称序列化为另外一个名称,如把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 Demo {  
  
    public static void main(String[] args) throws ParseException, IOException {  
        User user = new User();  
        user.setName("mask");   
        user.setEmail("mask@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);  
        //输出结果:{"name":"mask","birthday":"1996年09月30日","mail":"mask@sina.com"}  
    }  
}  

 转载:序列化和反序列化的详解_tree_ifconfig的博客-CSDN博客_序列化和反序列化

借鉴:序列化和反序列化理解 - w

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值