非静态内部类可访问外部类所有成员(含私有),静态内部类仅能访问外部类静态成员;外部类创建非静态内部类需先有实例;内部类静态成员只能是static final;匿名类无构造器、不能定义静态成员、只能继承一类或实现一接口;Lambda不是匿名类,不生成class文件,无法抛出未声明的检查异常,this指向外层类。
非静态内部类(即普通内部类)能直接访问外部类的所有成员,包括私有字段和方法;但静态内部类只能访问外部类的静态成员。这是由编译器生成的隐式引用决定的:每个非静态内部类实例都持有一个指向外部类实例的 this$0 引用。
non-static variable xxx cannot be referenced from a static context
new Inner() 创建非静态内部类实例,必须先有外部类实例,例如:new Outer().new Inner()
static final 常量(JVM 规范要求),否则编译失败Java 中的匿名类本质是省略了类名的子类声明,它没有构造函数,也不能定义静态成员,且只能继承一个类或实现一个接口(不能同时做两件事)。编译后会生成形如 Outer$1.class 的字节码文件。
new Thread(() -> {}) { ... } 是合法的,因为 Thread 有无参构造器;但 new SomeClass("required") { ... } 才能调用带参构造器final 或“事实 final”的局部变量(Java 8+ 放宽为“effectively final”,但语义不变)instanceof 判断匿名类是否属于某个具体类(它只有编译期类型,运行时类型是合成的),但可以判断是否实现了某接口虽然 Lambda 在语义上常用于替代单方法接口的匿名类(如 Runnable、Comparator),但它底层不生成独立 class 文件,也不继承任何类,而是通过 invokedynamic 指令绑定到函数式接口的抽象方法。因此它不具备匿名类的某些能力。
this 指向自身(它指向的是外层 enclosing 类的实例),而匿名类中的 this 指向的是匿名类实例本身非静态内部类默认持有对外部类实例的引用,因此要支持序列化,外部类也必须实现 Serializable,否则反序列化会失败并抛出 java.io.NotSerializableException。静态内部类则无此依赖。

static,避免意外捕获和序列化负担Thread、Socket),且内部类又间接引用了它们,仍可能在反序列化时失败InvalidClassException
public class Outer implements Serializable {
private String data = "outer";
private transient Thread unsafe = new Thread();
static class StaticInner implements Serializable {
void doWork() { /* 安全:不依赖 Outer 实例 */ }
}
class NonStaticInner implements Serializable {
void accessOuter() {
System.out.println(data); // OK
// System.out.println(unsafe); // 编译通过,但反序列化失败
}
}}
内部类和匿名类的关键差异不在语法糖,而在编译产物、生命周期绑定和序列化契约。很多 NPE 或 NotSerializableException 都源于没意识到那个隐式的 this$0 引用。