匿名内部类必须基于已有类型(接口或父类)声明,不能凭空创建,需实现接口或继承非final类,且只能访问final或effectively final的局部变量。

匿名内部类必须基于已有类型(接口或父类)声明
Java 中的匿名内部类不是独立类型,它必须显式继承一个类或实现一个接口。编译器会自动生成一个带编号的合成类(如 OuterClass$1.class),但你不能直接引用这个类名。常见误写是试图“凭空” new 一个匿名类:
new () { void doSomething() {} };这在 Java 中非法——编译器会报错 error: illegal start of expression。
正确写法必须有明确的上下文类型:
- 实现接口:
new Runnable() { public void run() { ... } } - 继承非 final 类:
new ArrayList(双括号初始化)() { { add("a"); } } - 不能继承 final 类(如
String)、不能实现多个接口、不能有显式构造器
访问外部变量时只允许使用 final 或 effectively final 变量
匿名内部类实例持有对外部作用域变量的隐式引用,为避免生命周期不一致,Java 要求这些变量必须是 final 或事实上不可再赋值(effectively final)。下面这段代码会编译失败:
int x = 10; new Thread(() -> System.out.println(x)).start(); x = 20; // 编译错误:local variables referenced from a lambda expression must be final or effectively final
注意:这个限制仅针对局部变量,对实例字段或静态字段无约束。另外,JDK 8+ 对“effectively final”的判定较宽松,只要变量在初始化后未被重新赋值即可,无需显式加 final 关键字。
立即学习“Java免费学习笔记(深入)”;
匿名内部类与 Lambda 表达式的本质区别
Lambda 表达式只能用于函数式接口(仅含一个抽象方法的接口),而匿名内部类可实现任意接口或继承类。这意味着:
-
new ActionListener() { public void actionPerformed(ActionEvent e) { ... } }✅ 可行 -
() -> { ... }✅ 可替代上面(因ActionListener是函数式接口) -
new AbstractList✅ 匿名子类可行() { public String get(int i) { return null; } public int size() { return 0; } } -
() -> { ... }❌ 无法替代抽象类继承,Lambda 不支持继承语法
性能上,Lambda 在多数场景下比匿名内部类更轻量(可能复用实例、避免生成额外 .class 文件),但调试时堆栈信息不如匿名类直观。
匿名内部类中 this 和 super 的指向容易混淆
在匿名内部类里,this 指向的是匿名类自身的实例,不是外部类实例;要访问外部类的成员,需用 OuterClassName.this.field。例如:
public class Outer {
private String name = "outer";
void method() {
new Thread(new Runnable() {
private String name = "inner";
public void run() {
System.out.println(this.name); // "inner"
System.out.println(Outer.this.name); // "outer"
}
}).start();
}
}
另一个易错点:匿名类无法直接调用外部类的私有构造器或私有方法(除非通过公有/包级方法间接暴露),且无法声明静态成员(包括静态字段、静态方法、静态初始化块)——编译器会报 illegal static declaration in inner class。










