关于java:为什么需要调用超类构造函数?

Why is it necessary to invoke a superclass constructor?

本问题已经有最佳答案,请猛点这里访问。

我目前正在学习Java中的继承,我理解它有很大的困难,但是我不能理解的一个主要原因是为什么在下面的例子中调用超类构造函数是必要的,它又有什么帮助呢?这个例子正好来自Oracle的super(...)方法教程。

1
2
3
4
5
6
7
public MountainBike(int startHeight,
                int startCadence,
                int startSpeed,
                int startGear) {
  super(startCadence, startSpeed, startGear);
  seatHeight = startHeight;
}

我最初的想法是,super(startCadence, startSpeed, startGear);将缩短当前构造函数中参数的代码,它将简单地获取父类中传递的值并将其添加到子类中(如下所示):

1
2
3
public MountainBike(int startHeight) {
 ...
}

然而,正如我已经知道的那样,这是错误的,再一次让我想到一个问题,那就是super(startCadence, startSpeed, startGear);到底做了什么。我会非常感谢某种形式的解释,这样一个初学者,像我一样,可以真正理解,而不是给我定义。


因为MountainBikeBike的子类,所以它应该可以在任何需要Bike对象的上下文中使用。如果不在子类中调用父类的构造函数,则无法保证子对象已正确初始化为充当父类的实例。

我模糊地记得,在Java中,如果您不给子构造函数中的父构造函数显式调用,则默认地调用父级的默认构造函数(即,不带参数)。在您的例子中,这还不够,因为或者没有默认的构造函数,或者您确实需要将给子构造函数的额外参数传递给父构造函数。


假设您的类MountainBike继承自Bike。单词super是对类的父类的引用。

如果Bike有这样的构造函数:

1
2
3
4
5
public Bike(int startCadence, int startSpeed, int startGear) {
    cadence = startCadence;
    speed = startSpeed;
    gear = startGear;
}

然后是MountainBike构造函数,它看起来如下:

1
2
3
4
public MountainBike(int startCadence, int startSpeed, int startGear, int startHeight) {
    super(startCadence, startSpeed, startGear);
    seatHeight = startHeight;
}

几乎等同于:

1
2
3
4
5
6
public MountainBike(int startCadence, int startSpeed, int startGear, int startHeight) {
    super.cadence = startCadence;
    super.speed = startSpeed;
    super.gear = startGear;
    seatHeight = startHeight;
}

除前者外,我们称之为super构造函数更好。首先,它避免了重复代码。对于两个,它保存两行。例如,如果我需要在speed上包含一些验证逻辑,我可以将其包含在Bike构造函数中,并且不需要在MountainBike构造函数中复制该代码。

切普纳的回答解释了为什么第二个版本会有问题的一些其他原因。但这个答案有助于解释对super构造函数的调用正在做什么。