今天刷题的时候遇到了一题关于内部类的访问的问题,发现自己忘了许多,现在来复习一下。
首先,内部类的类字节码文件名字为Outer$Inner.class。
记住一下访问原则:
1.内部类相当于外部类的一个成员变量,因此内部类可以直接访问外部类的成员。但是注意内部类静态方法不能访问外部类非静态成员
2.但是外部类如果要访问内部类中的成员,要么内部类是静态的,跟着外部类一起被加载;要么创建内部类的对象,通过对象访问(想让外部类进内部类里面,总得有内部类的“实体”吧,这个“实体”要么是跟外部类一起加载的静态内部类,要么是新建的内部类的对象,无论是哪种,都会在堆中为内部类开辟空间,然后才可以访问到)
例子:
a.外部类访问静态内部类的静态成员(一路畅通无阻)
Outer.Inner.func();
b.外部类访问静态内部类的非静态成员(一路畅通到内部类门口,
发现里面还没东西,怎么办?创建对象!)
//一般方式
Outer.Inner in = new Outer.Inner();
in.func();
//匿名方式
new Outer.Inner().func();//new Outer.Inner()创建出对象,然后调用对象的func
c.外部类访问非静态内部类的非静态成员(没对象,内部类的门都找不到!乖乖一步步创建对象吧)
//一般方式
Outer.Inner in = new Outer().new Inner();
in.func();
//匿名方式
new Outer().new Inner().func();
3.内部类如果有静态成员,则内部类也必须是静态的(否则内部类成员要随着外部类加载,发现内部类自己都还没加载,那这个成员是哪儿跑出来的呢?)
4.内部类访问局部变量时,局部变量必须加final修饰符。(随着方法运行完毕,局部变量会被释放,如果此时一个内部类还引用该变量则会报错。说到底就是变量的生命周期比内部类实例要短。而final保证这个变量始终指向一个对象,内部类就可以放心大胆地引用了。其实犯了和C++一样的毛病:返回了局部变量的引用)
注意:在JDK8版本之中,方法内部类中调用方法中的局部变量,可以不需要修饰为 final,匿名内部类也是一样的,主要是JDK8之后增加了 Effectively final 功能
class Outer{
public static void main(String[] args){
Object obj = new Outer().method();
}
Object method(){
int locvar = 1;
class Inner{
void displayLocvar(){
System.out.println("locvar = " + locvar);
}
}
Object in = new Inner();
return in;//返回了内部类对象,但是该对象依然引用着局部变量
}
}
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。