super是编译期确定的直接父类成员访问入口,不触发多态、不绕过访问控制;调用构造器须首行且唯一;访问字段和方法均为静态绑定;内部类中A.super特指外围类方法调用。

super 不是用来“调用父类方法”的万能钥匙,而是编译器用来定位当前类继承链中**紧邻父类**的成员访问入口。它不触发多态,也不绕过访问控制,更不是运行时动态查找机制。
super 调用构造方法必须是第一行
在子类构造器中,super() 或 super(...) 必须作为首条语句出现,否则编译失败。这是因为 JVM 要求对象初始化必须从最顶层父类开始逐层向下执行。
- 若子类构造器没写
super(...),编译器会自动插入无参super()—— 但前提是父类存在无参构造器,否则报错Implicit super constructor XXX() is undefined -
this(...)和super(...)不能共存于同一个构造器中 - 即使父类构造器是
private,只要子类在同一个类里(比如内部类),super()仍可调用;但普通继承下,父类构造器至少得是protected或包内可见
super 访问成员变量和方法不走虚方法表
用 super.methodName() 调用的是**编译期就确定的父类版本**,哪怕该方法在子类中被重写,也不会触发多态分派。
class A {
void foo() { System.out.println("A.foo"); }
}
class B extends A {
void foo() { System.out.println("B.foo"); }
void bar() {
foo(); // 输出 B.foo(动态绑定)
super.foo(); // 输出 A.foo(静态绑定)
}
}
-
super.field总是读取父类定义的字段,哪怕子类声明了同名字段(此时发生字段隐藏,而非覆盖) - 如果父类字段是
private,super.field在子类中不可见,编译报错,哪怕子类有同名字段也不行 - 接口中没有
super的使用场景,因为接口不能有构造器,且默认方法不能用super显式调用父接口(Java 8+ 接口间不构成继承链意义上的“父类”)
super 在内部类中可能指向外围类
当在一个非静态内部类(inner class)中,外部类也叫 A,而内部类又继承了某个类 B,这时 super 的含义取决于上下文:在构造器里是父类 B,但在方法里写 A.super.method() 是合法语法,表示调用外围类实例的 method()。
立即学习“Java免费学习笔记(深入)”;
-
A.this和A.super是两种不同机制:A.this获取外围类实例引用,A.super是 Java 为内部类特设的语法糖,用于显式调用外围类中被隐藏的方法(比如内部类重写了同名方法) - 这种
A.super.m()写法仅在内部类中有效,且要求A是直接外围类,不能跳级(如Outer2.super.m()在嵌套三层时无效) - 它和继承链里的
super完全无关,只是同名关键字被复用了
最容易被忽略的一点是:super 不是“向上找父类”,而是“向编译期确定的那个直接父类”。如果父类本身是泛型类、桥接方法介入、或者用了 lombok 的 @SuperBuilder,super 的实际绑定目标可能和直觉不符——这时候得看字节码里的 invokespecial 指令指向谁,而不是源码里写了什么。










