java中序列化与反序列化

一、基本概念

1、序列化和反序列化的定义:

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

    Java反序列化就是指把字节序列恢复为Java对象的过程。

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

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

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

2、json/xml的数据传递:

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

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

3、序列化优点:

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

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

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

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

① 将对象实例相关的类元数据输出。

② 递归地输出类的超类描述直到不再有超类。

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

④ 从上至下递归输出实例的数据。

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

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

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

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

  ①java.io.ObjectInputStream:对象输入流。

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

 ②java.io.ObjectOutputStream:对象输出流。

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

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

①若Student类仅仅实现了Serializable接口,则可以按照以下方式进行序列化和反序列化。

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

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

       ObjectOutputStream调用Student对象的writeObject(ObjectOutputStream out)的方法进行序列化。 
       ObjectInputStream会调用Student对象的readObject(ObjectInputStream in)的方法进行反序列化。

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

       ObjectOutputStream调用Student对象的writeExternal(ObjectOutput out))的方法进行序列化。 
       ObjectInputStream会调用Student对象的readExternal(ObjectInput in)的方法进行反序列化。

4、序列化和反序列化代码示例

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(“lihao”, “wjwlh”, “21”);
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;                                                 
}                                                                     

}
①序列化图示

②反序列化图示

三、序列化和反序列化的注意点:

①序列化时,只对对象的状态进行保存,而不管对象的方法;
在这里插入图片描述

②当一个父类实现序列化,子类自动实现序列化,不需要显式实现Serializable接口;
在这里插入图片描述

③当一个对象的实例变量引用其他对象,序列化该对象时也把引用对象进行序列化;

④并非所有的对象都可以序列化,至于为什么不可以,有很多原因了,比如:

安全方面的原因,比如一个对象拥有private,public等field,对于一个要传输的对象,比如写到文件,或者进行RMI传输等等,在序列化进行传输的过程中,这个对象的private等域是不受保护的;

资源分配方面的原因,比如socket,thread类,如果可以序列化,进行传输或者保存,也无法对他们进行重新的资源分配,而且,也是没有必要这样实现;

⑤声明为static和transient类型的成员数据不能被序列化。因为static代表类的状态,transient代表对象的临时数据。

⑥序列化运行时使用一个称为 serialVersionUID 的版本号与每个可序列化类相关联,该序列号在反序列化过程中用于验证序列化对象的发送者和接收者是否为该对象加载了与序列化兼容的类。为它赋予明确的值。显式地定义serialVersionUID有两种用途:

在某些场合,希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有相同的serialVersionUID;

在某些场合,不希望类的不同版本对序列化兼容,因此需要确保类的不同版本具有不同的serialVersionUID。

⑦Java有很多基础类已经实现了serializable接口,比如String,Vector等。但是也有一些没有实现serializable接口的;

⑧如果一个对象的成员变量是一个对象,那么这个对象的数据成员也会被保存!这是能用序列化解决深拷贝的重要原因;

注意:浅拷贝请使用Clone接口的原型模式。

  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java序列化是将对象转换为字节流的过程,以便将其存储在磁盘上或通过网络传输。而反序列化则是将字节流转换回对象的过程。 下面是一个示例,演示如何序列化反序列化一个Java对象: ```java import java.io.*; public class SerializationExample { public static void main(String[] args) { // 创建一个Person对象 Person person = new Person("Alice", 30); try { // 将Person对象序列化到文件 FileOutputStream fileOut = new FileOutputStream("person.ser"); ObjectOutputStream out = new ObjectOutputStream(fileOut); out.writeObject(person); out.close(); fileOut.close(); System.out.println("Person对象已经序列化到person.ser文件"); } catch (IOException e) { e.printStackTrace(); } try { // 将Person对象从文件反序列化出来 FileInputStream fileIn = new FileInputStream("person.ser"); ObjectInputStream in = new ObjectInputStream(fileIn); Person serializedPerson = (Person) in.readObject(); in.close(); fileIn.close(); System.out.println("从person.ser文件反序列化出的Person对象:"); System.out.println("姓名:" + serializedPerson.getName()); System.out.println("年龄:" + serializedPerson.getAge()); } catch (IOException e) { e.printStackTrace(); } catch (ClassNotFoundException e) { e.printStackTrace(); } } } class Person implements Serializable { private static final long serialVersionUID = 1L; private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } public String getName() { return name; } public int getAge() { return age; } } ``` 运行该程序,它会将一个Person对象序列化到文件person.ser,并从该文件反序列化出Person对象,并将其打印出来。 需要注意的是,要将一个对象序列化,该对象的类必须实现java.io.Serializable接口。否则,将会抛出NotSerializableException异常。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值