在Java中如何使用super关键字_Java父类调用解析

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.fieldNamesuper.methodName() 可明确访问父类版本。注意:这不适用于 private 成员,它们不可见;也不适用于静态方法——super.staticMethod() 虽然语法合法,但实际调用的是编译期确定的类型(即父类),而非运行时对象类型,容易造成误解。

关键区别:

  • 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) —— 这些都会编译失败。

真正表示当前对象的仍是 thissuper 只是告诉编译器:“这里我要找父类的那个东西”。运行时不存在两个对象,只有一个实例,只是视图不同。

容易忽略的一点:泛型桥接方法或 lambda 捕获中若涉及 super,可能触发意外的类型推导问题,此时建议显式转型或提取为局部变量。