inner class内部类
- 内部类
- 局部内部类
- 匿名内部类
- 静态内部类
- 代理
内部类(inner class
)是定义在另一个类中的类。使用内部类的原因有三点
- 内部类方法可以访问该类定义所在的作用域中的数据,包括私有的数据
- 内部类可以对同一个包中的其他类隐藏起来
- 想要定义一个回调函数且节省代码时,使用匿名(
anonymous
)内部类比较便捷
内部类默认没有定义构造器,所以编译器为这个类生成了一个默认的构造器,其代码如下所示:
public InnerClass(Object ob){
outer = object; //outer指外围对象的引用
}
当在start
方法中创建了InnerClass
对象后,编译器就会将this
引用传递给当前的构造器
InnerClass innerClass = new InnerClass(this);
只有内部类可以是私有类,而常规类只可以具有包可见性,或公有可见
对于内部类的外围类的引用outer
,其正规语法为
OuterClass.this
内部类是一种编译器现象,与虚拟机无关。编译器将会把内部类翻译成用$
分隔外部类名与内部类名的常规类文件,而虚拟机对此一无所知。
内部类可以访问外围类的私有数据,那么如何管理那些额外的访问特权呢?通过反射
class InnerClass{
static boolean access$0(Object);
}
我们看到,编译器在外围类添加静态方法access$0
,它将返回作为参数传递给它的对象域。
局部内部类
当某个类名字(例如Action
)只在某方法(例如run()
方法)中创建这个类型的对象时使用了一次,可以在这个方法中中定义局部类
public void run(){
class Action implement ActionListener {
//write this class
}
Action action = new Action();
action.start();
}
局部类不能用public
或private
访问说明符进行声明。它的作用域被限定在声明这个局部类的块中。它的优势,即对外部世界可以完全地隐藏起来。即使包含了run()
方法的类中的其他代码也不能访问它。除了run()
方法之外,没有任何方法知道Action
类的存在。
与其他内部类相比,局部类还有一个优点。他们不仅能够访问包含他们的外部类,还可以访问局部变量。不过,那些局部变量必须被声明为final
。
class Exam{