java继承序列化_Java中具有继承的对象序列化

在序列化中,当引入继承时,则根据超类和子类定义了某些情况,这使对每种情况下的序列化的理解变得更加简单。应遵循的基本规则如下。

1.当超类实现时,可序列化接口而子类则不。

在这种情况下,即使子类未实现Serializable接口,默认情况下,当超类被序列化时,子类的对象也将被序列化。

示例public class TestSerialization {

public static void main(String[] args) throws IOException, ClassNotFoundException {

B obj = new B();

FileOutputStream fos = new FileOutputStream("abc.ser");

ObjectOutputStream oos = new ObjectOutputStream(fos);

oos.writeObject(obj);

FileInputStream fis = new FileInputStream("abc.ser");

ObjectInputStream ois = new ObjectInputStream(fis);

B obj1 = (B)ois.readObject();

System.out.println(obj1.i + " " + obj1.j);

}

}

class A implements Serializable {

int i = 10;

}

class B extends A {

int j =20;

}

输出结果value of i is : 10 & value of j is : 20

2.当超类未实现可序列化的接口而子类实现时。

在这种情况下,在子类中继承的超类实例变量不会被序列化,并且在子类的序列化过程中也不会释放其分配的值。此外,在子类序列化期间的JVM也将默认的初始化值重新分配给这些超类的实例变量。在此场景中需要注意的一点是,超类必须具有默认的无参数构造函数,因为反序列化期间JVM访问超类。如果不存在该构造函数,则会遇到编译时异常。

示例public class TestSerialization {

public static void main(String[] args) throws IOException,ClassNotFoundException {

B obj = new B(10,20);

FileOutputStream fos = new FileOutputStream("abcd.ser");

ObjectOutputStream oos = new ObjectOutputStream(fos);

oos.writeObject(obj);

FileInputStream fis = new FileInputStream("abcd.ser");

ObjectInputStream ois = new ObjectInputStream(fis);

B obj1 = (B) ois.readObject();

System.out.println("value of i is : " +obj1.i + " & value of j is : " + obj1.j);

}

}

class A {

int i;

A() {

System.out.println("父类的默认构造函数。");

}

A(int i) {

this.i=i;

}

}

class B extends A implements Serializable {

int j;

B(int i , int j) {

super(i);

this.j=j;

}

}

输出1

存在默认构造函数时。父类的默认构造函数。

value of i is : 0 & value of j is : 20

输出2

不存在默认构造函数时。Exception in thread "main" java.io.InvalidClassException: B; B; no valid constructor

3.当需要序列化超类而不是子类时(自定义序列化)。

为了防止子类被序列化,我们需要实现writeObject()并readObject()在序列化和反序列化期间由JVM执行的方法以及这些方法抛出NotSerializableException。我们还可以在这些方法中提供自定义逻辑,这些逻辑将在序列化/去角质素。

示例public class TestSerialization {

public static void main(String[] args) throws IOException, ClassNotFoundException {

B obj = new B();

FileOutputStream fos = new FileOutputStream("abc.ser");

ObjectOutputStream oos = new ObjectOutputStream(fos);

oos.writeObject(obj);

FileInputStream fis = new FileInputStream("abc.ser");

ObjectInputStream ois = new ObjectInputStream(fis);

B obj1 = (B)ois.readObject();

System.out.println("value of i is : " + obj1.i + " & value of j is : " + obj1.j);

}

}

class A implements Serializable {

int i = 10;

}

class B extends A {

int j =20;

}

//实现writeObject方法,

private void writeObject(ObjectOutputStream out) throws IOException {

throw new NotSerializableException();

}

//实现readObject方法,

private void readObject(ObjectInputStream in) throws IOException {

throw new NotSerializableException();

}

输出结果Exception in thread "main" java.io.NotSerializableException

at B.writeObject(A.java:20)

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答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、付费专栏及课程。

余额充值