【IT168 文档】有时候开发时为了方便,在代码中使用了不少嵌套类(nested class),但是使用过程中如果不了解嵌套类的特性,可能会造成意想不到的情况。
Java对象想要通过网络传输,必须实现序列化接口,于是我们在Service中定义了了一个实现了序列化接口的内部类,然后调用远程接口将其传输到网络上。
一运行报异常:不能将没有实现序列化接口的Object序列化。
怎么回事,这是一个很简单的内部类,的确已经实现了序列化接口了,其定义的成员都是可序列化的String类型;将其换成普通类没有问题。难道不能使用序列化的内部类?
其实我们使用的内部类是嵌套类(nested class)的一种,而nested class 共有四种:static nested class 静态嵌套类
inner class 内部类(非静态)
local class 本地类(定义在方法内部)
anonymous class 匿名类
静态嵌套类的行为更接近普通的类,另外三个是真正的内部类。区别在于作用域的不同。
以下是对他们的性质描述:
以下还有一段程序说明它们的区别和使用方式:
Java代码
/***
* 普通内部类持有对外部类的一个引用, 静态内部类却没有
*
*@authorhoward*/publicclassOutterClass {/** this is static nested class*/privatestaticclassStaticNestedClass {privatevoidyell() {
System.out.println(this.toString());//OutterClass.this.yell();//静态内部类实例没有外部类实例的引用}
}/** this is inner class*/privateclassInnerClass {privatevoidyell() {
System.out.println(this.toString());
OutterClass.this.yell();//内部类实例显式使用外部类实例的方式}
}privatevoidyell() {
System.out.println(this.toString());
}privatevoidrun() {/** this is local class*/classLocalClass {publicvoidyell(){
System.out.println(this.toString());
}
}/** this is anonymous class*/newObject() {publicvoidyell(){
System.out.println(this.toString());
}
}.yell();
LocalClass lc=newLocalClass();
InnerClass ic=newInnerClass();
StaticNestedClass sc=newStaticNestedClass();
lc.yell();
ic.yell();
sc.yell();
}publicstaticvoidmain(String[] args) {
OutterClass oc=newOutterClass();
oc.run();
}
}
仔细分析如上代码,可以得出一个结论,所有的内部类,Local内部类,匿名内部类都可以直接访问外面的封装类的实例变量和方法。而静态嵌套类则不能。
调试代码可以发现,内部类,Local内部类,匿名内部类的实例都持有一个外部封装类实例的隐式引用;
而java对象序列化要求对象里所有的对象成员都实现序列化接口。
所以,如果只有内部类实现序列化,而外部封装类没有实现序列化接口,就会在对内部类进行序列化的时候报出异常。