Java中最终字段的继承?

The*_*tor 10 java inheritance static

当一个超类有一个标记为final的字段,但是一个子类覆盖(隐藏?)这个字段时会发生什么?"最终"并没有阻止这一点,是吗?我正在研究的具体例子是建筑类,不同类型的建筑物从中继承.除其他外,每种类型的成本应该是每个子类的最终成本,但每种类型的建筑物应该有自己的成本.

编辑:我已经意识到我不知道上面说的是什么.我真正想要的是成本的静态变量.但是,如果我在超类中声明这些静态变量,它们对于超类是静态的,因此Subclass1.cost例如引用与Superclass.cost或Subclass2.cost相同的值.如何创建对每个子类都是静态的变量,而不必在每个类中声明它们.

tem*_*def 16

final当应用于Java类的字段时,该关键字与继承无关.相反,它表示在构造函数之外,该字段无法重新分配.

Java分别处理名称隐藏和覆盖.覆盖实际上通过切换调用哪个函数来改变程序在运行时的可观察行为,而名称隐藏通过更改正在引用哪个字段的静态解释来更改程序. final应用于覆盖只适用于方法,因为Java中的字段无法覆盖.final不幸的是,在这些不同的上下文中的使用有点令人困惑,并且没有办法阻止字段将其名称隐藏在子类中.

如果您希望建筑物具有不同的成本,则可以选择使用重写getCost方法,该方法在每个派生类中被不同地覆盖.或者,您可以在基类中使用单个protectedprivate字段来存储成本,然后让每个子类直接设置此字段(如果是protected)或通过基类构造函数(如果此字段为private).

希望这可以帮助!

  • @TheTedinator-如果您放弃该字段的“ final”修饰符,它将起作用。如果该字段为“ final”,则不能在子类构造函数中为其分配不同的值。相反,子类构造函数必须将值传递给要分配的超类构造函数(如我在回答中所示)。如果超类未在其构造函数中分配值,则代码将无法编译;如果子类构造函数试图从超类设置`final`字段的值,则代码将无法编译。 (2认同)

Ted*_*opp 5

您可以通过两种方式执行此操作:创建访问器函数并隐藏字段本身,或者将构建成本从子类传递给超类构造函数。您可以组合这些并创建一个不能被子类覆盖的属性:

public class Building {
    private final int cost;

    protected Building(int cost, ...) {
        this.cost = cost;
    }

    public final int getCost() {
        return cost;
    }
}

public class Cottage extends Building {
    public Cottage() {
        super(COTTAGE_COST, ...);
    }
}
Run Code Online (Sandbox Code Playgroud)