IO/NIO — 对象序列化

本文详细介绍了Java中的对象序列化与反序列化过程,包括如何实现序列化接口、序列化和反序列化示例以及使用Transient关键字阻止变量被序列化。通过序列化,对象可以被保存到磁盘或在网络中传输,而反序列化则能恢复这些对象。Transient关键字允许开发者控制哪些变量不应包含在序列化过程中。
摘要由CSDN通过智能技术生成

对象序列化的目标是将对象保存在磁盘中,或者是在网络中可以传输对象

1、对象序列化(输出)

就是将内存上的Java对象转换成与平台无关的二进制流,可以将这种二进制流持久的保存在磁盘上,或者是可以在网络上传输

  • 实现

    实现接口Serializable或者Externalizable,该接口仅仅是标识一下可序列化,无需实现任何方法

2、对象序列化示例

将一个Java对象序列化成二进制流,输出到 test.txt 文件里面

public class WriteObject{
    public static void main(Stirng[] args) throws IOException{
        // 创建一个对象输出流
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("test.txt"));
        // new 一个对象
        Car car = new Car("红旗汽车");
        oos.writeObject(car);
    }
}
3、对象反序列化(输入)
  • 反序列化只读取Java对象的数据,不是Java类,在进行反序列化的时候,必须提供Java对象所属类的文件
  • 如果向文件序列化多个对象时,反序列化恢复时,也需要按照写入的顺序
  • 当对象时引用类型,进行反序列化时,这个引用的类也需要是能可序列化的,而且,该类如果有父类,为了反序列化时能正常恢复,该父类也会被序列化

注意:当引用类父子类关系重复时,也只会序列化一次,根据序列化编号区分(当类变化时,序列化编号也不会变)

4、反序列化示例

从上面生成的 test.txt 文件中,读取 Car 对象

public class ReadObject{
    public static void main(Stirng[] args) throws IOException{
        // 创建一个对象输出流
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("test.txt"));
        // 读取Car对象,知道类型,强转
        Car c = (Car)ois.readObject();
    }
}
5、Transient 关键字阻止变量被序列化
  • 在变量声明前加Transient关键字,可以阻止该变量被序列化到文件中,在反序列化时,Transient 变量的值被设为默认初始值(如int 的0,引用对象的null)
  • 例如传输密码等敏感数据时
### 回答1: Java 中的 Channel 对象本身不支持序列化,但是可以将 Channel 对象封装在可序列化对象中,并在需要的时候使用 Channel 对象。 比如,你可以定义一个可序列化的类,其中包含一个 Channel 对象的成员变量: ``` import java.io.Serializable; import java.nio.channels.Channel; public class SerializableChannel implements Serializable { private final Channel channel; public SerializableChannel(Channel channel) { this.channel = channel; } public Channel getChannel() { return channel; } } ``` 然后你就可以将 SerializableChannel 对象序列化并传输到另一个 JVM 进程中,并在需要的时候获取 Channel 对象: ``` SerializableChannel serializableChannel = ...; Channel channel = serializableChannel.getChannel(); ``` 注意,在传输 SerializableChannel 对象之后,Channel 对象可能不再与原来的通道连接,因此你需要根据你的需求确定是否需要重新打开通道。 ### 回答2: 在Java中,Channel对象是用于访问IO操作的实体,例如文件、网络连接等。由于Channel的设计目的是用于IO操作,它并不直接支持序列化接口。但是我们可以通过一些间接的方式来实现对Channel对象序列化。 一种常见的方法是使用包装类。我们可以创建一个自定义的类,将Channel对象作为该类的成员变量,并实现Serializable接口。通过这种方式,在序列化时,我们可以将该类对象作为一个整体进行传输,间接实现对Channel对象的传输。 另一种方法是使用外部化类。我们可以创建一个实现Externalizable接口的类,在类中定义与Channel对象相关的属性和方法,并在序列化和反序列化时,手动存储和读取Channel对象的状态。这样我们可以控制序列化和反序列化的过程,将Channel中的关键信息存储和恢复。 除了以上两种简单的方法,还可以利用第三方库来实现对Channel对象序列化。例如,可以使用Google的Protobuf库或Apache的Avro库来定义和序列化Channel对象的数据结构,然后将序列化后的字节流进行传输或持久化。 总之,由于Channel对象的设计初衷是用于IO操作,它本身并不直接支持序列化接口。但我们可以通过以上提到的方式,间接地实现对Channel对象序列化,方便在分布式系统中进行传输和持久化操作。 ### 回答3: 在Java中,channel对象是通过实现SelectableChannel接口来创建的,而SelectableChannel接口实际上是继承了Channel接口和Closeable接口。可序列化的接口是Serializable接口,因此要使Channel对象序列化,需要满足以下几个步骤: 1. 创建自定义的Channel类,该类需要实现SelectableChannel接口,并且需要实现Serializable接口。 ```java public class MyChannel implements SelectableChannel, Serializable { // 实现SelectableChannel接口和Serializable接口的方法 // ... } ``` 2. 序列化操作时,首先需要确保该类的所有成员变量都是可序列化的。如果有非序列化的成员变量,可以使用transient关键字进行修饰,使其不参与序列化。 ```java public class MyChannel implements SelectableChannel, Serializable { private transient int id; // 不参与序列化 // 需要序列化的其他成员变量 // ... } ``` 3. 如果父类也实现了Serializable接口,需要调用父类的序列化方法、字段以及构造函数。 ```java public class MyChannel extends SelectableChannel implements Serializable { public MyChannel() { super(); // 调用父类的构造函数 } private void writeObject(java.io.ObjectOutputStream out) throws IOException { // 调用父类的序列化方法 out.defaultWriteObject(); // 其他序列化操作 // ... } private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException { // 调用父类的反序列化方法 in.defaultReadObject(); // 其他反序列化操作 // ... } } ``` 通过以上步骤,就能够在Java中实现将Channel对象序列化。但需要注意的是,序列化并不会序列化channel的状态,仅用于传输对象。在对象序列化后,需要重新进行channel的状态配置。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值