关于Java内部类序列化

情况一:

public class DistributeData implements  SerializedName{

 public class CodeDetail {...}

}

CodeDetail并不会被序列化。

情况二:

public class DistributeData implements  SerializedName{

 public class CodeDetail implements  SerializedName{...}

}

报NotSerializableException,查错误,CodeDetail这个类虽然实现了Serializable接口,但CodeDetail在项目中是以内部类的形式定义的,

而内部类不能被序列化!

对于内部类来说,静态的内部类才可以被序列化。应该是:

public class DistributeData implements  SerializedName{

 public static class CodeDetail implements  SerializedName{...}

}

因此,尽量少用内部类,限制太多,以免造成不必要的麻烦。

情况三:

public class DistributeData {

 public class CodeDetail implements  SerializedName{...}

}

运行报异常:不能将没有实现序列化接口的Object序列化。
所有的内部类,Local内部类,匿名内部类都可以直接访问外面的封装类的实例变量和方法。而静态嵌套类则不能。
内部类,Local内部类,匿名内部类的实例都持有一个外部封装类实例的隐式引用,而java对象序列化要求对象里所有的对象成员都实现序列化接口。所以,如果只有内部类实现序列化,而外部封装类没有实现序列化接口,就会在对内部类进行序列化的时候报出异常。

原理解释:

内部类是嵌套类(nested class)的一种,而nested class 共有四种:

static nested class 静态嵌套类
inner class 内部类(非静态)
local class 本地类(定义在方法内部)
anonymous class 匿名类
  静态嵌套类的行为更接近普通的类,另外三个是真正的内部类。区别在于作用域的不同。
这里写图片描述

http://www.jianshu.com/p/8b8bfe1fd488

  • 2
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
部类序列过程和普通类是一样的。但是,需要注意的是,如果内部类是非静态的,则需要序列其外部类的引用,以便在反序列时能够正确地重建内部类实例。同时,如果内部类实现了 Serializable 接口,则需要确保其外部类也实现了 Serializable 接口,否则在序列时会抛出 NotSerializableException 异常。 以下是一个示例代码: ```java import java.io.*; class Outer implements Serializable { private static final long serialVersionUID = 1L; private int outerValue; public Outer(int outerValue) { this.outerValue = outerValue; } public int getOuterValue() { return outerValue; } public void setOuterValue(int outerValue) { this.outerValue = outerValue; } class Inner implements Serializable { private static final long serialVersionUID = 1L; private int innerValue; public Inner(int innerValue) { this.innerValue = innerValue; } public int getInnerValue() { return innerValue; } public void setInnerValue(int innerValue) { this.innerValue = innerValue; } } } public class SerializationExample { public static void main(String[] args) throws Exception { Outer outer = new Outer(10); Outer.Inner inner = outer.new Inner(20); // 序列 FileOutputStream fos = new FileOutputStream("data.ser"); ObjectOutputStream oos = new ObjectOutputStream(fos); oos.writeObject(outer); oos.close(); // 反序列 FileInputStream fis = new FileInputStream("data.ser"); ObjectInputStream ois = new ObjectInputStream(fis); Outer outer2 = (Outer) ois.readObject(); ois.close(); // 验证序列结果 System.out.println("Outer value: " + outer2.getOuterValue()); System.out.println("Inner value: " + outer2.new Inner(0).getInnerValue()); } } ``` 在上面的示例中,我们定义了一个包含内部类的外部类 Outer,并进行了序列和反序列。注意,在序列和反序列时,我们都需要将外部类和内部类同时序列和反序列。最终输出的结果如下: ``` Outer value: 10 Inner value: 20 ``` 这表明内部类序列和反序列过程都是成功的。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值