Java中的继承机制允许子类复用父类的属性和方法,同时可以扩展自身独有的功能,是实现代码复用和多态的基础。通过继承,我们可以构建出层次化的类结构,让代码的逻辑更加清晰。
Java继承的基础语法
在Java中实现继承需要使用extends关键字,语法格式为class 子类 extends 父类。需要注意Java只支持单继承,一个子类只能直接继承一个父类,但是父类还可以被其他类继承,形成继承链。
以下是一个简单的继承示例:
// 定义父类
class Parent {
// 父类成员变量
String name = "父类名称";
// 父类方法
public void sayHello() {
System.out.println("父类打招呼方法");
}
}
// 子类继承父类
class Child extends Parent {
// 子类独有的成员变量
int age = 10;
// 子类独有的方法
public void showAge() {
System.out.println("子类年龄:" + age);
}
}
public class Test {
public static void main(String[] args) {
Child child = new Child();
// 子类可以调用父类的方法
child.sayHello();
// 子类可以访问父类的成员变量
System.out.println(child.name);
// 调用子类独有的方法
child.showAge();
}
}
继承的底层实现原理
类元数据的关联
当Java虚拟机加载类时,会为每个类生成对应的类元数据,存储在方法区中。子类的类元数据中会记录其直接父类的引用,通过这个引用,子类可以找到父类的所有可继承的成员信息,包括方法、字段等。这也是子类能够使用父类成员的基础。
成员访问规则
子类继承父类时,并不是把父类的成员复制到自己的内存空间中,而是在访问时通过父类的引用去查找。具体规则如下:
- 父类中用
private修饰的私有成员,子类无法直接访问,但是可以通过父类提供的public或protected方法间接访问。 - 父类中用
default修饰的默认成员,只有同包下的子类可以访问。 - 父类中用
protected和public修饰的成员,子类可以直接访问。
方法重写与动态绑定
如果子类定义了和父类方法签名完全相同的方法,就会发生方法重写。此时子类对象调用该方法时,会优先执行子类重写后的方法,这个过程依赖Java的动态绑定机制。
动态绑定发生在程序运行时,虚拟机根据对象的实际类型来确定要调用的方法,而不是根据引用变量的类型。我们可以通过下面的代码理解这个过程:
class Animal {
public void makeSound() {
System.out.println("动物发出声音");
}
}
class Dog extends Animal {
// 重写父类的makeSound方法
@Override
public void makeSound() {
System.out.println("狗汪汪叫");
}
}
class Cat extends Animal {
// 重写父类的makeSound方法
@Override
public void makeSound() {
System.out.println("猫喵喵叫");
}
}
public class DynamicBindTest {
public static void main(String[] args) {
Animal animal1 = new Dog();
Animal animal2 = new Cat();
// 运行时根据实际对象类型调用对应方法
animal1.makeSound(); // 输出:狗汪汪叫
animal2.makeSound(); // 输出:猫喵喵叫
}
}
super关键字的作用
super关键字用来在子类中访问父类的成员,或者调用父类的构造方法。常见的使用场景有以下几种:
- 调用父类的构造方法:子类构造方法的第一行默认会隐式调用
super(),也就是父类的无参构造方法,如果需要调用父类的有参构造,需要显式写出super(参数)。 - 访问父类的成员变量:当子类和父类有同名的成员变量时,用
super.变量名可以访问父类的变量。 - 调用父类的方法:当子类重写了父类的方法,又需要调用父类原方法时,可以用
super.方法名()。
下面是一个super关键字的使用示例:
class Base {
String msg = "父类消息";
public Base(String msg) {
this.msg = msg;
}
public void printMsg() {
System.out.println("父类打印:" + msg);
}
}
class Derived extends Base {
String msg = "子类消息";
public Derived(String baseMsg) {
// 调用父类的有参构造方法
super(baseMsg);
}
@Override
public void printMsg() {
// 调用父类的printMsg方法
super.printMsg();
// 访问子类的msg
System.out.println("子类打印:" + msg);
// 访问父类的msg
System.out.println("通过super访问父类msg:" + super.msg);
}
}
public class SuperTest {
public static void main(String[] args) {
Derived derived = new Derived("传入父类的消息");
derived.printMsg();
}
}
继承中的构造方法执行顺序
当创建子类对象时,构造方法的执行顺序是先执行父类的构造方法,再执行子类的构造方法。如果继承链有多层,会从最顶层的父类开始依次向下执行构造方法。这个顺序保证了父类的成员在子类使用之前已经被正确初始化。
我们可以通过下面的多层继承示例验证这个顺序:
class GrandParent {
public GrandParent() {
System.out.println("祖父类构造方法执行");
}
}
class Parent extends GrandParent {
public Parent() {
System.out.println("父类构造方法执行");
}
}
class Child extends Parent {
public Child() {
System.out.println("子类构造方法执行");
}
}
public class ConstructorOrderTest {
public static void main(String[] args) {
Child child = new Child();
}
}
运行上述代码,输出顺序为:祖父类构造方法执行、父类构造方法执行、子类构造方法执行,符合我们提到的执行规则。
继承的注意事项
- 不要滥用继承,只有当两个类确实存在is-a的关系时,才适合使用继承,比如狗是动物的一种,适合让Dog继承Animal。
- 子类重写父类方法时,访问权限不能比父类的方法更严格,比如父类方法是public,子类重写的方法不能是protected或private。
- 父类的构造方法不能被继承,只能通过super关键字在子类构造方法中调用。
- 如果父类没有无参构造方法,子类必须显式调用父类的有参构造方法,否则编译会报错。
Java继承类继承原理extends关键字方法重写super关键字修改时间:2026-06-09 18:12:33