Java序列化与反序列化

1. 序列化的时候必须实现Serializable这个接口,基本类型的数据像Integer,String这些都实现了Serializable这个接口,实现序列化的时候有两种方式

  a. 实现Serializable这个接口,如果有不需要序列化的属性使用transient修饰

 b. 实现externalizable这个接口,复习里面的方法,根据需求去 实现要序列化和反序列化的字段和属性,externalizable是Serializable的子类;实现这个接口需要重写writeExternal和readExternal方法,指定对象序列化的属性和从序列化文件中读取对象属性的行为

c. 在用Serializable这个接口在实现序列化的时候,其父类必须要有个无参数的构造函数,否则反序列化就会报错;默认如果没有父类的话,自定义的类的类的父类为Object类;如下代码,反序列化时则会报错:

package Serializable;

import net.bytebuddy.implementation.bind.annotation.Super;

import java.io.*;

class Anninal    {
    private int type;
    public Anninal(int type){
      System.out.println("I will be Seriz in Anni");
    }
}

public class Person extends Anninal implements  Serializable {
    private transient String name;
    private int age;
    

    //我不提供无参构造器
    public Person(String name, int age) {
        super(10);
        System.out.println("I will be Person");
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

 class WriteObject {
    public static void main(String[] args) {
        try (//创建一个ObjectOutputStream输出流
             ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("abcd.txt"))) {
            //将对象序列化到文件s
            Person person = new Person("9龙", 23);
            oos.writeObject(person);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}


 class ReadObject {
    public static void main(String[] args) {
        try (//创建一个ObjectInputStream输入流
              ObjectInputStream ois = new ObjectInputStream(new FileInputStream("abcd.txt"))) {
            Person brady = (Person) ois.readObject();
            System.out.println(brady);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

报错如下:

java.io.InvalidClassException: Serializable.Person; no valid constructor
    at java.io.ObjectStreamClass$ExceptionInfo.newInvalidClassException(ObjectStreamClass.java:150)
    at java.io.ObjectStreamClass.checkDeserialize(ObjectStreamClass.java:768)
    at java.io.ObjectInputStream.readOrdinaryObject(ObjectInputStream.java:1775)
    at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1351)
    at java.io.ObjectInputStream.readObject(ObjectInputStream.java:371)
    at Serializable.ReadObject.main(Person.java:57)

因为父类Anninal没有默认的构造函数,修改代码如下,则正确

package Serializable;

import net.bytebuddy.implementation.bind.annotation.Super;

import java.io.*;
//implements  Serializable
class Anninal    {
    private int type;
    public Anninal(int type){
      System.out.println("I will be Seriz in Anni");
    }

    public Anninal(){
        System.out.println("I will be Seriz in Anninal, I am a Default Constructor");
    }
}

public class Person extends Anninal implements  Serializable {
    private transient String name;
    private int age;


    //我不提供无参构造器
    public Person(String name, int age) {
        super(10);
        System.out.println("I will be Person");
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

 class WriteObject {
    public static void main(String[] args) {
        try (//创建一个ObjectOutputStream输出流
             ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("abcd.txt"))) {
            //将对象序列化到文件s
            Person person = new Person("9龙", 23);
            oos.writeObject(person);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}


 class ReadObject {
    public static void main(String[] args) {
        try (//创建一个ObjectInputStream输入流
              ObjectInputStream ois = new ObjectInputStream(new FileInputStream("abcd.txt"))) {
            Person brady = (Person) ois.readObject();
            System.out.println(brady);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
输出结果如下:

I will be Seriz in Anninal, I am a Default Constructor
Person{name='null', age=23}

如果父类实现Serializable,父类没有默认的够造函数也可以
package Serializable;

import net.bytebuddy.implementation.bind.annotation.Super;

import java.io.*;
class Anninal  implements  Serializable  {
    private int type;
    public Anninal(int type){
      System.out.println("I will be Seriz in Anni");
    }

}

public class Person extends Anninal implements  Serializable {
    private transient String name;
    private int age;


    //我不提供无参构造器
    public Person(String name, int age) {
        super(10);
        System.out.println("I will be Person");
        this.name = name;
        this.age = age;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

 class WriteObject {
    public static void main(String[] args) {
        try (//创建一个ObjectOutputStream输出流
             ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("abcd.txt"))) {
            //将对象序列化到文件s
            Person person = new Person("9龙", 23);
            oos.writeObject(person);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}


 class ReadObject {
    public static void main(String[] args) {
        try (//创建一个ObjectInputStream输入流
              ObjectInputStream ois = new ObjectInputStream(new FileInputStream("abcd.txt"))) {
            Person brady = (Person) ois.readObject();
            System.out.println(brady);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
这个则不会报错

private static final long serialVersionUID 的序列化版本号,只有版本号相同,即使更改了序列化属性,对象也可以正确被反序列化回来

如果反序列化使用的class的版本号与序列化时使用的不一致,反序列化会报InvalidClassException异常。

序列化版本号可自由指定,如果不指定,JVM会根据类信息自己计算一个版本号,这样随着class的升级,就无法正确反序列化;不指定版本号另一个明显隐患是,不利于jvm间的移植,可能class文件没有更改,但不同jvm可能计算的规则不一样,这样也会导致无法反序列化

什么情况下需要修改serialVersionUID呢?分三种情况。

  • 如果只是修改了方法,反序列化不容影响,则无需修改版本号;
  • 如果只是修改了静态变量,瞬态变量(transient修饰的变量),反序列化不受影响,无需修改版本号;
  • 如果修改了非瞬态变量,则可能导致反序列化失败。如果新类中实例变量的类型与序列化时类的类型不一致,则会反序列化失败,这时候需要更改serialVersionUID。如果只是新增了实例变量,则反序列化回来新增的是默认值;如果减少了实例变量,反序列化时会忽略掉减少的实例变量
  1. 所有需要网络传输的对象都需要实现序列化接口,通过建议所有的javaBean都实现Serializable接口。
  2. 对象的类名、实例变量(包括基本类型,数组,对其他对象的引用)都会被序列化;方法、类变量、transient实例变量都不会被序列化。
  3. 如果想让某个变量不被序列化,使用transient修饰。
  4. 序列化对象的引用类型成员变量,也必须是可序列化的,否则,会报错。
  5. 反序列化时必须有序列化对象的class文件。
  6. 当通过文件、网络来读取序列化后的对象时,必须按照实际写入的顺序读取。
  7. 单例类序列化,需要重写readResolve()方法;否则会破坏单例原则。
  8. 同一对象序列化多次,只有第一次序列化为二进制流,以后都只是保存序列化编号,不会重复序列化。
  9. 建议所有可序列化的类加上serialVersionUID 版本号,方便项目升级。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java序列化是将Java对象转换为字节序列的过程,以便将其存储在文件中或通过网络传输。反序列化则是将字节序列转换回Java对象的过程。 Java提供了两种序列化机制:Java序列化和外部化(Externalization)序列化。其中Java序列化是自动进行的,而外部化序列化需要手动实现。 Java序列化的实现方式是通过将Java对象转换为字节流,然后将字节流写入文件或通过网络发送。反序列化则是从字节流中读取数据并将其还原为Java对象。 序列化反序列化的步骤如下: 序列化: 1.创建一个对象输出流(ObjectOutputStream)。 2.调用对象输出流的writeObject()方法将Java对象转换为字节流并写入文件或网络。 反序列化: 1.创建一个对象输入流(ObjectInputStream)。 2.从输入流中读取字节流并调用readObject()方法将其转换为Java对象。 需要注意的是,被序列化的对象必须实现Serializable接口,该接口没有任何方法,只是标识该类可以被序列化。 以下是一个Java序列化反序列化的示例代码: ``` import java.io.*; public class SerializationDemo { public static void main(String[] args) { // 序列化对象 try { // 创建一个Student对象 Student s = new Student("张三", 18, "男"); // 创建一个对象输出流 ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("student.ser")); // 将Student对象写入输出流中 oos.writeObject(s); // 关闭输出流 oos.close(); } catch (IOException e) { e.printStackTrace(); } // 反序列化对象 try { // 创建一个对象输入流 ObjectInputStream ois = new ObjectInputStream(new FileInputStream("student.ser")); // 从输入流中读取Student对象 Student s = (Student) ois.readObject(); // 输出Student对象的属性 System.out.println(s.getName()); System.out.println(s.getAge()); System.out.println(s.getGender()); // 关闭输入流 ois.close(); } catch (IOException | ClassNotFoundException e) { e.printStackTrace(); } } } // Student类需要实现Serializable接口 class Student implements Serializable { private String name; private int age; private String gender; public Student(String name, int age, String gender) { this.name = name; this.age = age; this.gender = gender; } public String getName() { return name; } public int getAge() { return age; } public String getGender() { return gender; } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值