super()必须位于子类构造方法首行,否则编译错误;super.成员用于访问被隐藏的父类非private成员,不可在static上下文或赋值使用。

super() 必须是子类构造方法的第一行
Java 规定,如果子类构造方法中显式调用 super() 或 super(…),它必须出现在第一行;否则编译器会自动插入无参的 super()。一旦你写了任何其他语句(比如 System.out.println() 或变量赋值),再写 super() 就会报错:call to super must be first statement in constructor。
常见错误场景:
- 想在调用父类构造前做参数校验 —— 改用静态工厂方法或提取校验逻辑到单独的
private static方法 - 误以为可以用
this()和super()共存 —— 二者互斥,只能选其一,且都必须是首行 - 父类没有无参构造,子类又没写
super(...)—— 编译失败,提示constructor Parent() is undefined
super.成员名 用于访问被子类隐藏的父类成员
当子类定义了与父类同名的字段或方法(非重写),用 super.fieldName 或 super.methodName() 可明确访问父类版本。注意:这不适用于 private 成员,它们不可见;也不适用于静态方法——super.staticMethod() 虽然语法合法,但实际调用的是编译期确定的类型(即父类),而非运行时对象类型,容易造成误解。
关键区别:
立即学习“Java免费学习笔记(深入)”;
-
this.field:优先取子类定义的字段(字段隐藏,不是重写) -
super.field:强制读取父类中同名字段(前提是protected或包内可见) -
super.method():调用父类版本的方法(可用于在重写方法中复用父逻辑)
class Parent {
protected String name = "Parent";
void print() { System.out.println("Parent.print"); }
}
class Child extends Parent {
String name = "Child"; // 隐藏父类字段
void show() {
System.out.println(this.name); // Child
System.out.println(super.name); // Parent
super.print(); // Parent.print
}
}
super 在重写方法中调用父类逻辑很常见但有陷阱
重写方法里用 super.methodName() 是合理且推荐的做法,尤其在模板模式或需要扩展行为时。但要注意两点:父类方法是否为 final(不能调)、是否抛出受检异常(子类重写方法需兼容声明)。
典型误用:
- 在
static方法中写super.xxx()—— 编译错误,super不可用于静态上下文 - 在构造方法外、实例方法内调用
super()—— 语法错误,super()只能在构造方法首行出现 - 重写
toString()或equals()时漏掉super.equals()导致父类字段未参与比较
super 不是引用,不能赋值给变量或用于 instanceof
super 不是一个对象引用,也不是关键字意义上的“值”,它只是编译器提供的访问父类成员的语法通道。所以你不能写 Object obj = super;,也不能写 if (super instanceof Parent) —— 这些都会编译失败。
真正表示当前对象的仍是 this,super 只是告诉编译器:“这里我要找父类的那个东西”。运行时不存在两个对象,只有一个实例,只是视图不同。
容易忽略的一点:泛型桥接方法或 lambda 捕获中若涉及 super,可能触发意外的类型推导问题,此时建议显式转型或提取为局部变量。










