Java里extends关键字怎么用_Java继承语法规则说明

extends只能用于类继承,即class SubClass extends SuperClass;不能用于接口定义或实例创建,子类单继承且须调用父类构造器,继承public/protected成员但受访问控制限制,建立is-a关系并支持动态绑定。

extends只能用于类继承,不能用于接口或实例

Java 中 extends 的唯一合法用途是让一个**类**显式继承另一个**类**,语法为 class SubClass extends SuperClass。它不能写在接口定义里(接口用 extends 是另一套规则,但那是接口继承接口),也不能写在对象创建或变量声明处——比如 new ArrayList() extends List 这种写法完全非法。

常见错误现象:

  • 误在接口中用 class MyInterface extends Runnable → 编译报错:「class expected」
  • 误在方法返回类型后加 extends → 如 String extends CharSequence method() → 语法错误

子类只能单继承,且必须调用父类构造器

Java 不支持多类继承,一个类最多有一个直接父类。即使不显式写 extends,所有类也默认隐式继承 Object。而一旦显式 extends,编译器会强制要求子类构造器第一行必须调用父类构造器(通过 super(...)),否则报错 Implicit super constructor Xxx() is undefined

实操建议:

  • 若父类只有带参构造器,子类构造器必须显式写 super(arg1, arg2)
  • 若忘了写 super() 且父类无无参构造器,JDK 直接拒绝编译
  • this(...)super(...) 不能共存于同一构造器中

被继承的成员受访问修饰符限制

子类能继承父类的 publicprotected 成员(包括字段、方法、嵌套类),但无法直接访问 private 成员,也无法继承 default(包级私有)成员(除非子类与父类在同一包)。

容易踩的坑:

  • 父类有个 private String name,子类写 System.out.println(name) → 编译失败
  • 父类方法是 void print() { ... }(即 default 访问级别),子类在不同包中重写该方法 → 编译不通过,因为根本不可见
  • 子类重写父类方法时,访问权限不能比父类更严格(如父类是 protected,子类不能改为 private

继承不是复制,而是建立“is-a”关系和运行时类型链

extends 建立的是编译期静态继承关系,但更重要的是它决定了 JVM 的运行时行为:子类实例可以赋值给父类引用(向上转型),能通过 instanceof 检查,且虚方法调用走动态绑定。

Animal a = new Dog(); // 合法:Dog is-a Animal
a.speak(); // 实际调用 Dog.speak(),非 Animal.speak()
System.out.println(a instanceof Dog); // true

性能影响很小,但设计上必须意识到:继承带来强耦合。过度使用 extends(比如只为复用代码而继承)常导致脆弱的类层次,此时应优先考虑组合(has-a)或接口(implements)。

真正难处理的,往往是继承链中多个 super 调用顺序、构造器参数传递路径,以及父类方法被意外重写后引发的逻辑漂移——这些不会报错,但会让行为变得难以追踪。