廖雪峰Java6 IO编程-2input和output-7序列化

1.序列化

序列化是指把一个Java对象变成二进制内容byte[]

  • 序列化后可以把byte[]保存到文件中
  • 序列化后可以把byte[]通过网络传输
  • 一个Java对象要能序列化,必须实现Serializable接口:
    * Serializable接口没有定义任何方法
    * 空接口被称为标记接口(Marker Interface)
  • ObjectOutputStream负责把一个Java对象写入二进制流:
try(ObjectOutputStream output = new ObjectOutputStream(...)){
    output.writeObject(new Person("小明"));
    output.writeObject(new Person("小红"));
}

2.反序列化

反序列化是指把一个二进制内容(byte[])变成Java对象

  • 反序列化后可以从文件读取byte[]并变成Java对象
  • 反序列化后可以从网络读取byte[]并变为Java对象
  • ObjectInputStream负责从二进制流读取一个Java对象
try(ObjectInputStream input = new ObjectInputStream(...)){
    Object p1 = input.readObject();//使用readObject()获取Object对象
    Person p2 = (Person) input.readObject();//如果读取的Person类型,需要强制转型
}

readObject()可能抛出的异常:

  • ClassNotFoundException:没有找到对应的Class。反序列化的Java程序没有Person类,不能反序列化
  • InvalidClassException:Class不匹配。序列化的Person的age是int,反序列化的Person的age是long类型

    反序列化的重要特点:
  • 反序列化由JVM直接构造出Java对西那个,不调用构造方法,即构造方法的初始化代码根本不执行。

    3.示例

    Person.java

import java.io.Serializable;

public class Person implements Serializable {
    private String name;
    public Person(String name){
        System.out.println(name+"诞生了");
        this.name = name;
    }
    public String getName(){
        return name;
    }
    public void setName(String name){
        this.name = name;
    }
}

Main.java

public class Main {
    public static void main(String[] args) throws IOException,ClassNotFoundException {
       String dataFile = "saved.data";
       try(ObjectOutputStream output = new ObjectOutputStream(
               new BufferedOutputStream(
                       new FileOutputStream(dataFile)))){
           //依次写入int,String,Person对象
           output.writeInt(999);
           output.writeUTF("Hello world");
           output.writeObject(new Person("小明"));
           output.writeObject(new Person("小红"));
       }
        System.out.println("Read...");
       try(ObjectInputStream input = new ObjectInputStream(
               new BufferedInputStream(
                       new FileInputStream(dataFile)))){
           //依次读取int,String,Person对象
           System.out.println(input.readInt());
           System.out.println(input.readUTF());
           Person p1 = (Person) input.readObject();//构造方法不会执行
           System.out.println(p1);
           Person p2 = (Person) input.readObject();
           System.out.println(p2);
       }
    }
}

4. serialVersionUID

为Person添加serialVersionUID,Main操作1次。
Person中serialVersionUID发生改变,只保留读取,再次执行Main,结果报错
Person.java

import java.io.Serializable;

public class Person implements Serializable {
    public static final long serialVersionUID = 1234567890123456L;//Main执行1次后将UID修改为1234567890L
    private String name;
    public Person(String name){
        System.out.println(name+"诞生了");
        this.name = name;
    }
    public String getName(){
        return name;
    }
    public void setName(String name){
        this.name = name;
    }
}
public class Main {
    public static void main(String[] args) throws IOException,ClassNotFoundException {
       String dataFile = "saved.data";
        //修改UID后,注释写入代码块
       try(ObjectOutputStream output = new ObjectOutputStream(
               new BufferedOutputStream(
                       new FileOutputStream(dataFile)))){
           //依次写入int,String,Person对象
           output.writeInt(999);
           output.writeUTF("Hello world");
           output.writeObject(new Person("小明"));
           output.writeObject(new Person("小红"));
       }
        System.out.println("Read...");
       try(ObjectInputStream input = new ObjectInputStream(
               new BufferedInputStream(
                       new FileInputStream(dataFile)))){
           //依次读取int,String,Person对象
           System.out.println(input.readInt());
           System.out.println(input.readUTF());
           Person p1 = (Person) input.readObject();//构造方法不会执行
           System.out.println(p1);
           Person p2 = (Person) input.readObject();
           System.out.println(p2);
       }
    }
}

5.总结:

  • 可序列化的Java对西那个必须实现java.io.Serializable接口
  • 类似Serializable这样的空接口被称为标记接口(Marker Interface)
  • 反序列化时不调用构造方法
  • 可设置serialVersionUID作为版本号(非必需)
  • Java的序列化机制仅适用于Java,如果需要与其他语言交换数据,必须使用通用的序列化方法,例如JSON。

原文地址:https://www.cnblogs.com/csj2018/p/10661999.html

时间: 2024-10-31 14:38:21

廖雪峰Java6 IO编程-2input和output-7序列化的相关文章

廖雪峰Java6 IO编程-2input和output-4Filter模式

1.JDK提供的InputStream分为两类: 直接提供数据的InputStream * FileInputStream:从文件读取 * ServletInputStream:从HTTP请求读取数据 * Socket.getInputStream():从TCP连接读取数据 提供额外附加功能的FilterInputStream * 如果要给FileInputStream添加缓冲功能: BufferedFileInputStream extends FileInputStream * 如果要给Fi

廖雪峰Java13网络编程-1Socket编程-2TCP编程

在开发网络应用程序的时候,会遇到Socket这个概念. Socket是一个抽象概念,一个应用程序通过一个Socket来建立一个远程连接,而Socket内部通过TCP/IP协议把数据传输到网络. Socket/TCP/部分IP都是由操作系统提供的.不同的编程语言只是提供了对操作系统调用的加单的封装,例如Java提供的几个Socket相关的类就封装了操作系统提供的接口. 为什么需要Socket? 因为仅仅通过IP地址进行通信还不够,同一台计算机同一时间会运行多个网络程序.当计算机收到一个数据包的时候

廖雪峰Java13网络编程-1Socket编程-3TCP多线程编程

TCP多线程编程 一个ServerSocket可以和多个客户端同时建立连接,所以一个Server可以同时与多个客户端建立好的Socket进行双向通信. 因此服务器端,当我们打开一个Socket以后,通常使用一个无限for循环,在这个for循环内部,每次调用accept方法,返回一个与远程客户新建的Socket连接,紧接着启动一个新的线程,来处理这个连接. ServerSocket ss = new ServerSocket(port); for( ; ; ){ Socket sock = ss.

廖雪峰Java2面向对象编程-3继承和多态-1继承

1.继承 继承是一种代码复用的方式. Student与Person有相同部分的代码. Student可以从Person继承,这样Student获得了Person的所有功能,只需要编写新增的功能即可.通过继承,可以实现代码的复用. 继承使用关键字extends,一个类只能有一个父类. 如果没有写明继承类,编译器会自动指定该类继承于基类Object. Person:超类super,父类,基类 Student:子类subclass,扩展类 Person.java //默认继承Object public

廖雪峰Java2面向对象编程-4抽象类和接口-1抽象类

每个子类都可以覆写父类的方法 如果父类的方法没有实际意义,能否去掉方法的执行语句?子类会报编译错误 如果去掉父类的方法,就失去了多态的特性 可以把父类的方法声明为抽象方法. 如果一个class定义了方法,但没有具体执行代码,这个方法就是抽象方法: 抽象方法用abstract修饰 抽象方法没有任何执行语句 因为无法执行抽象方法,因此这个类也必须声明为抽象类abstract class 无法实例化一个抽象类.如果子类不是抽象类,依旧可以被实例化. 抽象类作用: 抽象类用于被继承 抽象类可以强迫子类实

廖雪峰Java2面向对象编程-4抽象类和接口-2接口

抽象方法本质上是定义接口规范 public abstract class Person{ public abstract void run(); } public class Student extends Person{ @Override public void run(){} } public class Teacher extends Person{ @Override public void run(){} } 如果一个抽象类没有字段,所有方法全部是抽象方法,就可以把抽象类改写为接口i

廖雪峰Java11多线程编程-3高级concurrent包-5Atomic

Atomic java.util.concurrent.atomic提供了一组原子类型操作: 如AtomicInteger提供了 int addAndGet(int delta) int incrementAndGet() int get() int compareAndGet() Atomic类可以实现: 无锁(lock-free)实现的线程安全(thread-safe)访问 原理:CAS(Compare and Set)如果AtomicInteger实例的值是prev,就替换为next,返回

廖雪峰Java16函数式编程-2Stream-1Stream简介

1. Stream Java8引入全新的Stream API 位于java.util.stream包 1.1 Stream API不同于java.io的InputStream/OutputStream java.io java.util.stream 存储 顺序读写的byte/char 顺序输出的任一Java对象 用途 序列化数据至文件/网络 内存计算/业务逻辑 1.2 Stream和List也是不同的 List元素已经分配并存储在内存中的,而Stream输出的元素并没有预先存储在内存中,它通常

廖雪峰Java2面向对象编程-2数据封装-1方法重载

方法重载 方法重载Overload是指:多个方法的方法名相同,但各自的参数不同 参数的个数不同 参数的类型不同 参数位置不同 方法返回值类型通常都是相同的 目的:相同功能的方法使用同一名字,便于调用 如以indexOf()为例 public static void main(String[] args) { String s = "Test"; int n1 = s.indexOf('e');//返回字符的索引 int n2 = s.indexOf("st");//