Loading

【Java基础】内部类

非静态内部类不能拥有静态变量 为什么

下面这段代码,如果Lazyholder没有static修饰,则编译不过

class Singleton2 {
    private static class LazyHolder {
        private static Singleton2 singleton;
    }
    public Singleton2 getInstance() {
        return LazyHolder.singleton;
    }
}

"非静态内部类不能拥有静态变量" ,可以这样理解,因为JVM类加载的顺序为:

 

java根据需要在运行时把字节码载入到内存,它分三个步骤:

1、加载:类加载器查找到字节码(.class文件)并根据这些字节码创建一个Class对象;

2、链接:验证类中的字节码,为静态域分配存储空间,需要的话同时解析这个类其它类的所有引用;

3、初始化:当类的静态方法(构造器是特殊的静态方法)或者非常数静态域(即不是编译器常量)被首次引用时,执行静态初始化块和初始化静态数据。

然后才能执行对象创建

 

如果LazyHolder没有static,则本质上,它其实是Singleton2的非静态成员,则他必须在Singleton2对象创建之后才会进行加载,然而问题在于,另一方面根据JVM的加载顺序,static 的singleton变量又必须在LazyHolder对象存在之前完成加载,这就矛盾了。

 

靠,上面的是昨天写的,今天突然看不懂了。我想了想,更简单的理解是:

      静态变量是要能够直接用类名.静态变量的方式来访问的,在上面这个例子中,实际上就是Singleton2.LazyHolder.singleton,是这样去访问,但是如果LazyHolder不是静态类,那问题就来了,你会发现singleton无法被访问了!因为Singleton.LazyHolder就已经不合法了,根本没法往下走。嗯,我觉得我自己的理解比网上查到的靠谱多了

 

什么是匿名内部类

实际上匿名内部类是对下面这种写法的简化

public class InnerClassLearn {
    class MyThread extends Thread {
        private int i = 11;
        public int value() {return i;}
    }
    public Thread thread() {return new MyThread();}

    public static void main(String[] args) {
        Thread thread = new InnerClassLearn().thread();
    }
}

用匿名内部类的方式重写上面的代码:

public class InnerClassLearn {
    
    public Thread thread() {
        return new Thread() {
            private int i = 11;
            public int value() {return i;}
        };
    }
    
    public static void main(String[] args) {
        Thread thread = new InnerClassLearn().thread();
    }
}

其中红色部分就是一个继承于 Thread类的匿名内部类

posted @ 2016-08-08 16:51  注销111  阅读(1798)  评论(0编辑  收藏  举报