Java 类中不仅可以定义变量和方法,还可以定义类,这样定义在类内部的类就被称为内部类。根 据定义的方式不同,内部类分为静态内部类,成员内部类,局部内部类,匿名内部类四种。
1. 静态内部类
public class Out1 {
private static int a;
private int b;
public static class Inner1 {
public void print() {
System.out.println(Out1.a);
System.out.println(new Out1().b);
}
}
public static void main(String[] args) {
Out1.Inner1 inner1 = new Out1.Inner1();
inner1.print();
}
}
- 静态内部类可以访问外部类所有的静态变量和方法,即使是 private 的也一样。
- 静态内部类和一般类一致,可以定义静态变量、方法,构造方法等。
- 其它类使用静态内部类需要使用“外部类.静态内部类”方式,如下所示:Out.Inner inner = new Out.Inner();inner.print();
- Java 集合类 HashMap 内部就有一个静态内部类 Entry。Entry 是 HashMap 存放元素的抽象,HashMap 内部维护 Entry 数组用了存放元素,但是 Entry 对使用者是透明的。像这种和外部类关系密切的,且不依赖外部类实例的,都可以使用静态内部类。
2. 成员内部类
public class Out2 {
private static int a;
private int b;
public class Inner2 {
public void print() {
System.out.println(Out2.a);
System.out.println(Out2.this.b);
}
}
public static void main(String[] args) {
Out2.Inner2 inner2 = new Out2().new Inner2();
inner2.print();
}
}
定义在类内部的非静态类,就是成员内部类。成员内部类不能定义静态方法和变量(final 修饰的 除外)。这是因为成员内部类是非静态的,类初始化的时候先初始化静态成员,如果允许成员内 部类定义静态变量,那么成员内部类的静态变量初始化顺序是有歧义的。
3. 局部内部类
public class Out3 {
private static int a;
private int b;
public Out3() {
}
public void test(final int c) {
int d = true;
class Inner3 {
Inner3(final int val$c) {
this.val$c = var2;
}
public void print() {
System.out.println(this.val$c);
}
}
}
}
定义在方法中的类,就是局部类。如果一个类只在某个方法中使用,则可以考虑使用局部类。
4. 匿名内部类
interface Print {
void print();
}
public class Out4 {
private static int a;
private int b;
public void print() {
new Print() {
@Override
public void print() {
System.out.println(Out4.a);
System.out.println(Out4.this.b);
}
}.print();
}
}
- 要继承一个父类或者实现一个接口、直接使用 new 来生成一个对象的引用
- 匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有 class 关键字,这是因为匿名内部类是直接使用 new 来生成一个对象的引
看一个案例
public class DotThis {
void f() {
System.out.println("DotThis.f()");
}
public class Inner {
public DotThis outer() {
return DotThis.this;
}
}
public Inner inner() {
return new Inner();
}
public static void main(String[] args) {
DotThis dt = new DotThis();
DotThis.Inner dti = dt.inner();
dti.outer().f();
}
}