IO流——ObjectInputStream和ObjectOutputStream

本文详细介绍了如何在Java中使用ObjectInputStream进行对象反序列化,以及ObjectOutputStream进行序列化操作。涉及序列化和反序列化的概念,以及Dog类的实现与使用示例。特别强调了对象序列化的要求和注意事项。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本次博客我们来学习IO流中的对象流——ObjectInputStream和ObjectOutputStream。

序列化和反序列化

  1. 序列化就是在保存数据时,保存数据的值和数据类型。
  2. 反序列化就是在恢复数据时,恢复数据的值和数据类型。
  3. 需要让某个对象支持序列化机制,则必须让其类是可序列化的,为了让某个类是可序列化的,该类必须实现如下两个接口之一:
    • Serializable //这是一个标记接口,没有方法。
    • Externalizable //该接口有方法需要实现,因此我们一般实现上面的 Serializable接口。

ObjectOutputStream

  • ObjectOutputStream 提供 序列化的功能
    请添加图片描述
  • 演示ObjectOutputStream的使用,完成数据的序列化。使用ObjectOutputStream 序列化 基本数据类型和 一个 Dog对象(name ,age) ,并保存到data,dat文件中。
public class ObjectOutputStream_ {
    public static void main(String[] args) throws IOException {
        //序列化后,保存的文件格式,不是纯文本,而是按照他的格式来保存
        String filePath = "d:\\111\\data.dat";

        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(filePath));

        //序列化数据到 d:\111\data.dat
        oos.write(100); //int ->Integer (实现了 Serializable)
        oos.writeBoolean(true); //boolean -> Boolean (实现了 Serializable)
        oos.writeChar('a'); //char -> Character (实现了 Serializable)
        oos.writeDouble(9.5); //double -> Double (实现了 Serializable)
        oos.writeUTF("领导"); //String

        //保存一个dog对象
        oos.writeObject(new Dog("来福",6));
        oos.close();
        System.out.println("数据保存完毕(序列化形式)");
    }
}

//如果需要序列化某个类的对象,实现Serializable
class Dog implements Serializable{
    private String name;
    private int age;
    public Dog(String name,int age){
        this.name = name;
        this.age = age;
    }
}

ObjectInputStream

  • ObjectInputStream 提供 反序列化功能
    请添加图片描述

  • 使用ObjectInputStream 读取 data.dat 并反序列化恢复数据。

    1. 如果我们希望调用Dog的方法,需要向下转型。

    2. 需要我们将Dog类的定义,拷贝到可以引用的位置。

public class Dog implements Serializable {
    private String name;
    private int age;
    public Dog(String name,int age){
        this.name = name;
        this.age = age;
    }

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

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}
public class ObjectInputStream_ {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        //指定反序列化的文件
        String filePath = "d:\\111\\data.dat";

        ObjectInputStream ois = new ObjectInputStream(new FileInputStream(filePath));

        //1.读取(反序列化)的顺序需要和你保存数据(序列化)的顺序一致,否则会出现异常
        System.out.println(ois.readInt());
        System.out.println(ois.readBoolean());
        System.out.println(ois.readChar());
        System.out.println(ois.readDouble());
        System.out.println(ois.readUTF());
        Object dog = ois.readObject();
        System.out.println("运行类型= "+dog.getClass());
        System.out.println("dog信息= "+dog); //底层 Object ->Dog

        //1. 如果我们希望调用Dog的方法,需要向下转型
        //2. 需要我们将Dog类的定义,拷贝到可以引用的位置。
        Dog dog1 = (Dog) dog;
        System.out.println(dog1.getName());

        //关闭流,关闭外层流即可,底层会关闭 FileInputStream
        ois.close();
    }
}

注意事项和细节说明

  1. 读写顺序要一致,序列化和反序列化的顺序要一致。
  2. 要求序列化或反序列化对象,需要实现Serializable。
  3. 序列化的类中建议添加SerialVersionUID,为了提高版本的兼容性。
  4. 序列化对象时,默认将里面所有属性都进行序列化, 但除了static或transient修饰的成员
  5. 序列化对象时,要求里面属性的类型也需要实现序列化接口。
  6. 序列化具备可继承性,也就是如果某类已经实现了序列化,则它的所有子类叶已经默认实现了序列化。
03-20
### Java IO概述 Java IO(Input/Output)用于处理数据输入输出操作。它提供了多种类来实现文件、网络其他设备的数据传输功能。这些类主要位于 `java.io` 包中。 #### 文件读写的两种基本方式 Java 中的 IO 分为两大类:字节字符。 - **字节**:适用于二进制数据的操作,例如图片、音频等文件。 - **字符**:专门针对文本文件设计,提供更方便的编码转换支持。 以下是常见的几种及其用途: | 类型 | 输入 | 输出 | |--------------|----------------------------------|-------------------------------| | 字节 | `InputStream` | `OutputStream` | | 字符 | `Reader` | `Writer` | --- ### 示例代码展示 #### 1. 使用字节复制图片文件 下面是一个完整的例子,展示了如何通过字节完成图片文件的复制[^2]。 ```java import java.io.*; public class ByteStreamExample { public static void main(String[] args) { try (InputStream is = new FileInputStream("source_image.png"); OutputStream os = new FileOutputStream("destination_image.png")) { byte[] buffer = new byte[1024]; int length; while ((length = is.read(buffer)) > 0) { os.write(buffer, 0, length); } } catch (IOException e) { System.err.println("Error occurred during file copy: " + e.getMessage()); } } } ``` 此代码片段实现了从源路径到目标路径的文件拷贝过程。其中使用了缓冲区技术以提高性能。 #### 2. 缓冲字节的应用 为了进一步提升效率,可以引入带缓冲机制的子类——`BufferedInputStream` `BufferedOutputStream`[^3]。 ```java import java.io.*; public class BufferedByteStreamExample { public static void main(String[] args) { try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream("input.txt")); BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("output.txt"))) { int data; while ((data = bis.read()) != -1) { bos.write(data); } } catch (IOException e) { System.err.println("I/O error encountered: " + e.getMessage()); } } } ``` 上述程序利用缓冲特性减少了底层磁盘访问次数,从而加快了执行速度。 #### 3. 对象序列化与反序列化 如果需要保存复杂结构的对象至外部存储介质,则需要用到对象 `ObjectInputStream` `ObjectOutputStream`[^1]。 ```java // 序列化对象 try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("object.ser"))) { MySerializableClass obj = new MySerializableClass(); oos.writeObject(obj); } catch (Exception ex) { System.out.println(ex.toString()); } // 反序列化对象 try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("object.ser"))) { MySerializableClass restoredObj = (MySerializableClass) ois.readObject(); System.out.println(restoredObj); } catch (Exception ex) { System.out.println(ex.toString()); } ``` 注意:要使某个类能够被序列化,该类需实现 `java.io.Serializable` 接口。 --- ### 总结 以上介绍了三种典型场景下的 Java IO 应用实例,分别是基于字节的基础文件操作、借助缓存优化后的高效版本以及面向对象持久化的高级技巧。每种情况都体现了不同层次的需求满足能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值