匿名内部类是Java中“定义即实例化”的语法糖,用于满足单次使用的接口/抽象类契约,适用于非函数式接口、需调用父类构造器、定义字段或方法等Lambda无法处理的场景。匿名内部类是 Java 中一种「定义即实例化」的语法糖,它没有类名,只能用一次,本质是编译器自动生成的子类或接口实现类(如
OuterClass$1.class)。
它不是炫技工具,而是为了解决“这个逻辑只用一次,但又必须满足接口/抽象类契约”的现实问题——比如点一下按钮、跑一个线程、排一次序。写成独立类反而冗余,用 Lambda 又受限(比如需要访问非 final 局部变量,或需调用父类构造器),这时候匿名内部类就是最直接的解法。
Lambda 表达式只能用于函数式接口(仅含一个抽象方法),且无法处理以下真实场景:
new Thread(Runnable) 可以用 Lambda,但 new TimerTask()(非函数式接口,有多个 public 方法)不行,只能用匿名内部类new AbstractList() { ... } 必须传参初始化,Lambda 无法做到new Comparator() { private int count = 0; @Override public int compare(...) { return ...; } }
WindowListener),你只关心 windowOpened,但必须实现全部 7 个方法——这时匿名内部类配合空实现更清晰(比 Lambda 强制全写强)Swing/AWT 中写 button.addActionListener(new ActionListener() { ... }) 看似简单,但容易忽略三点:
int i = 0;,编译直接报错 —— 不是运行时异常,是语法拒绝SwingUtilities.invokeLater(...) 容易漏:Swing 是单线程模型,所有 UI 更新必须在 Event Dispatch Thread,否则界面卡死或不刷新同理,new Thread(new Runnable() { ... }).start() 常见错误是忘记捕获异常:run() 方法内抛出未检查异常会静默终止线程

Thread.setDefaultUncaughtExceptionHandler。
当你要按多字段动态排序,又不想暴露完整策略类时,匿名内部类比 Lambda 更灵活:
Collections.sort(employees, new Comparator() { @Override public int compare(Employee a, Employee b) { int nameCmp = a.getName().compareTo(b.getName()); if (nameCmp != 0) return nameCmp; return Integer.compare(a.getAge(), b.getAge()); // 支持 null-safe 处理 } });
null(比如 a.getName() 可能为 null),就得拆成多行,可读性反而下降new Comparator() {...} 的类型信息在运行时已丢失,不要依赖它做反射判断