Java里super关键字的作用是什么_Java父类访问机制说明

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;
    }
}

s

uper.方法名() 用于在重写中复用父类逻辑

这不是“调用父类实例”,而是调用当前对象上被子类覆盖前的父类版本方法。典型场景是「增强而非替代」:

  • 常见于 toString()equals()、事件回调(如 onCreate())等重写方法中
  • 必须确保父类方法不是 privatefinal,否则编译不通过
  • 不能在静态方法或静态上下文中使用 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 显式访问父类字段:

  • 这种写法较少见,通常建议避免同名字段——它容易引发理解混乱和维护风险
  • 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 到底指向谁”,说明可能混淆了继承模型和对象内存布局——它不分配新空间,只是告诉编译器:“请从继承链上一层找这个成员”。