再来说说Nested Class

有时候开发时为了方便,在代码中使用了不少嵌套类(nested class),但是使用过程中如果不了解嵌套类的特性,可能会造成意想不到的情况。

Java对象想要通过网络传输,必须实现序列化接口,于是我们在Service中定义了了一个实现了序列化接口的内部类,然后调用远程接口将其传输到网络上。

 

一运行报异常:不能将没有实现序列化接口的Object序列化。

怎么回事,这是一个很简单的内部类,的确已经实现了序列化接口了,其定义的成员都是可序列化的String类型;将其换成普通类没有问题。难道不能使用序列化的内部类?

 

 

其实我们使用的内部类是嵌套类(nested class)的一种,而

nested class 共有四种:

  • static nested class  静态嵌套类
  • inner class              内部类(非静态)
  • local class               本地类(定义在方法内部)
  • anonymous class    匿名类

静态嵌套类的行为更接近普通的类,另外三个是真正的内部类。区别在于作用域的不同。

以下是对他们的性质描述:

 

Types of Nested Classes
TypeScopeInner
static nested classmemberno
inner [non-static] classmemberyes
local classlocalyes
anonymous classonly the point where it is definedyes

 

以下还有一段程序说明它们的区别和使用方式:

 

/**
 *
 * 普通内部类持有对外部类的一个引用, 静态内部类却没有
 *
 * @author howard
 */
public class OutterClass {

    /*
     * this is static nested class
     */
    private static class StaticNestedClass {
        private void yell() {
            System.out.println(this.toString());
            // OutterClass.this.yell();//静态内部类实例没有外部类实例的引用
        }
    }

    /*
     * this is inner class
     */
    private class InnerClass {
        private void yell() {
            System.out.println(this.toString());
            OutterClass.this.yell();//内部类实例显式使用外部类实例的方式
        }
    }

    private void yell() {
        System.out.println( this.toString());
    }

    private void run() {
        /*
         * this is local class
         */
        class LocalClass {
            public void yell(){
                System.out.println(this.toString());
            }
        }
        /*
         * this is anonymous class
         */
        new Object() {
            public void yell(){
                System.out.println(this.toString());
            }
        }.yell();
        LocalClass lc=new LocalClass();
        InnerClass ic = new InnerClass();
        StaticNestedClass sc=new StaticNestedClass();
        lc.yell();
        ic.yell();
        sc.yell();
    }

    public static void main(String[] args) {
        OutterClass oc = new OutterClass();
        oc.run();
    }
}

 

仔细分析如上代码,可以得出一个结论,所有的内部类,Local内部类,匿名内部类都可以直接访问外面的封装类的实例变量和方法。而静态嵌套类则不能。

 

调试代码可以发现,内部类,Local内部类,匿名内部类的实例都持有一个外部封装类实例的隐式引用;

而java对象序列化要求对象里所有的对象成员都实现序列化接口。

 

所以,如果只有内部类实现序列化,而外部封装类没有实现序列化接口,就会在对内部类进行序列化的时候报出异常。

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值