super的核心作用是让子类安全访问父类成员,它指向当前对象中继承自父类的部分;必须用于构造方法首行调用父类初始化,且不能与this()共存;可调用父类方法、访问非private字段,但非对象引用。
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;
}
}
逻辑这不是“调用父类实例”,而是调用当前对象上被子类覆盖前的父类版本方法。典型场景是「增强而非替代」:
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("狗可以跑和走"); // ✅ 再扩展
}
}当子类定义了与父类同名的非 private 字段时,该字段会**隐藏(hide)** 父类字段,而不是覆盖。此时直接写 name 访问的是子类字段,要用 super.name 显式访问父类字段:
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 到底指向谁”,说明可能混淆了继承模型和对象内存布局——它不分配新空间,只是告诉编译器:“请从继承链上一层找这个成员”。