内部类---静态内部类、非静态内部类(成员内部类、局部类、 匿名内部类)的解析和区别

外部顶级类的类名需和类文件名相同,只能使用public和default。而内部类是指在外部类的内部再定义一个类,类名不需要和文件名相同。内部类可以是静态static或者非静态的,也可用public,default(包限定),protected和private修饰。

内部类:

概念:

我们所说的内部类,官方的叫法是嵌套类(Nested Classes)。嵌套类包括静态内部类(Static Nested Classes)和内部类(Inner Classes)。而内部类分为成员内部类,局部内部类(Local Classes)和匿名内部类(Anonymous Classes)。

记住:静态的东西在第一次运行时就会被首先初始化(优先于对象的初始化new),所以它不能去调用非静态的东西

内部类是一个编译是的概念,一旦编译成功,就会成为完全不同的两个类,分别为outer.class和outer$inner.class类。所以内部类的成员变量/方法名可以和外部类的相同。

内部类的作用

1.内部类可以很好的实现隐藏
2.一般的非内部类,是不允许有 private 与protected权限的,但内部类可以
3.非静态内部类拥有外围类的所有元素的访问权限 (private修饰也能访问)
4.可是实现多重继承 (让多个内部类分别继承多个其他类,使外部类可以同时获取多个其他类的属性)
5.可以避免修改接口而实现同一个类中两种同名方法的调用。(外部类继承,让内部类实现接口)

一、静态内部类:

形如:

public class OuterClass {
    private String name;

    static class StaticInerCls{
        private String name;
    }
}

静态内部类除了访问权限修饰符比外围类多以外, 和外围类没有区别, 只是代码上将静态内部类组织在了外部类里面,但是注意静态内部类是不能访问外部类非静态变量及方法的。
创建静态内部类:以Class.Iner的形式

OuterClass.StaticInerCls staticInerCls = new OuterClass.StaticInerCls();  

二、非静态内部类:

非静态内部类能访问外部类的一切成员, 包括私有成员。外部类虽然不能直接访问内部类的成员, 但是可以通过内部类的实例访问内部类的私有成员。

1.1、成员内部类:

形如:

public class OuterCls {
    private String name;
    public String getName(){
        return name;
    }

    class InerCls{
        private String name;

        public String getName(){
            return name;
        }
    }

}

成员内部类可以直接使用外部类的所有成员和方法,即使是private修饰的。而外部类要访问内部类的所有成员变量和方法,内需要通过内部类的对象来获取。要注意的是,成员内部类不能含有static的变量和方法。因为成员内部类需要先创建了外部类,才能创建它自己。我们可以这样理解,非静态内部类都没有生成,它的静态方法、属性自然也就不能被创建出来
创建内部类对象方法,以object.new Iner的形式:

OuterCls outerCls = new OuterCls();
OuterCls.InerCls inerCls = outerCls.new InerCls();  

成员内部类不能有static修饰的成员,但是却允许定义常量。

public class OuterClass {
    private String name;

    static class StaticInerCls{
        private String name;
    }

    class InerCls{
        private String name;
        private static int id;    //不允许,会报错
        private static final int TYPE = 0;   //允许
    }
}

1.2、局部内部类:

指内部类定义在方法体内,只能在该方法或条件的作用域内才能使用,退出这写作用域就无法引用。

public class OuterCls {
    private String name;
    public String getName(){
        return name;
    }

    public void getInerCls(){
        class InerCls{
            private String name;

            public String getName(){
                return name;
            }
        }
    }

}

作为非静态内部类的一种特殊形式, 非静态内部类的所有限制对局部类同样成立。局部类不仅可以访问外部类的所有成员,还可以访问方法体的局部变量,但必须是final修饰的局部变量。

为什么局部类访问局部变量,变量必须加上final?

场景:

    public static void main(String[] args){
        Outer out = new Outer();
        Object obj = out.method();
    }

    Object method(){
        int localVariable = 0;
        class Inner{
            void println(){
                System.out.println("localVariable " + localVariable);
            }
        }
        Object in = new Inner();
        return in;
    }
}

这里的localVariable会变红,提示需要给localVariable变量加final修饰。
解析:这是作用域的问题。在方法method执行完成后,局部变量value就失效了,而在new Inner()产生的in对象还存在obj的引用,这样对象就访问了一个不存在的变量,是不允许的。iner还存在,在外面和后续调用该局部变量时,这个局部变量可能已经失效了。但为什么加上final就可以保证能访问呢?这里Java采用了一种copy local variable的方法实现,定义为final的变量,会拷贝一份存到局部内部类中,后续使用持续维护这个对象在生命周期内,所以可以继续访问。

注:final可以修饰的范围有类,方法,属性。修饰类,该类不可以被继承;修饰方法,该方法不可以被子类重写;修饰变量,该变量值不能被修改。

我们可以这样理解,在类中的方法执行完后就会被移出内存,这个方法里面的存放于栈内存的局部变量自然就失效了,但留下来new出来的存放于堆内存的对象,但是对象用到了那个局部变量,这就产生了尴尬,怎么解决呢,copy嘛,把这个变量和他的值复制到内部类内部,注意,这是通过复制放入内部类的,原数据是没进来的,这就是为什么有上面的那段话,说它不能再被改变,我们想想,人家都把数据复制存起来放别的地方了,如果原始数据又改变了,而我又不能及时去跟新,那我不就之前复制的数据是错了的嘛。

1.3、匿名内部类:

为了免去给内部类命名,或者只想使用一次,就可以选择使用匿名内部类。它的能力跟前面几种非静态内部类是相似的,只不过呢,没有类名,也就不能new出对象了,也就是说,我本来就不打算之后还用它

    public void countDown(){
        new Thread(){
            @Override
            public void run() {

            }
        }.start();


参考链接:https://www.jianshu.com/p/f0fdea957792

猜你喜欢

转载自blog.csdn.net/qq_36528311/article/details/87883120