super的核心作用是让子类安全访问父类成员,它指向当前对象中继承自父类的部分;必须用于构造方法首行调用父类初始化,且不能与this()共存;可调用父类方法、访问非private字段,但非对象引用。

super 的核心作用是让子类能明确、安全地访问父类的成员(构造方法、方法、字段),它不是“调用父类”,而是“指向当前对象中继承自父类的那一部分”。没有 super,当子类重写方法或隐藏字段时,就无法再触达父类原始逻辑。
super() 必须出现在子类构造方法第一行
创建子类对象时,JVM 要求父类部分必须先完成初始化。因此 super() 或 super(参数) 必须是子类构造方法的第一条语句:
- 若父类只有带参构造器(比如
Parent(String name)),子类构造器中**必须显式写**super(name),否则编译失败 - 若没写任何
super(...),编译器会自动插入super()—— 但前提是父类存在无参构造;否则报错Constructor Parent() is undefined -
super()和this()不能共存于同一个构造器首行,二者只能选其一
class Person {
String name;
Person(String name) { this.name = name; }
}
class Student extends Person {
int id;
Student(String name, int id) {
super(name); // ✅ 正确:显式调用父类有参构造
this.id = id;
}
}super.方法名() 用于在重写中复用父类逻辑
这不是“调用父类实例”,而是调用当前对象上被子类覆盖前的父类版本方法。典型场景是「增强而非替代」:
- 常见于
toString()、equals()、事件回调(如onCreate())等重写方法中 - 必须确保父类方法不是
private或final,否则编译不通过 - 不能在静态方法或静态上下文中使用
super,会报错non-static variable super cannot be referenced from a static context
class Animal {
void move() { System.out.println("动物可以移动"); }
}
class Dog extends Animal {
@Override
void move() {
super.move(); // ✅ 先执行父类逻辑
System.out.println("狗可以跑和走"); // ✅ 再扩展
}
}super.字段名 用于区分同名字段的归属
当子类定义了与父类同名的非 private 字段时,该字段会**隐藏(hide)** 父类字段,而不是覆盖。此时直接写 name 访问的是子类字段,要用 super.name 显式访问父类字段:
立即学习“Java免费学习笔记(深入)”;
- 这种写法较少见,通常建议避免同名字段——它容易引发理解混乱和维护风险
-
super只能访问protected、包内可见或public字段;对private字段无效(编译报错) - 字段访问不涉及多态,
super.name在编译期就绑定到父类字段,和运行时类型无关
class Base {
String name = "Base";
}
class Derived extends Base {
String name = "Derived";
void print() {
System.out.println(name); // 输出 "Derived"
System.out.println(super.name); // 输出 "Base"
}
}最容易被忽略的一点是:super 不是一个对象引用,也不可赋值给变量(比如 Object s = super; 是非法的)。它只是一种语法符号,代表“当前对象的父类视图”。一旦你开始纠结“super 到底指向谁”,说明可能混淆了继承模型和对象内存布局——它不分配新空间,只是告诉编译器:“请从继承链上一层找这个成员”。










