Java:当超级构造函数需要参数时如何初始化子级

时间:2012-08-02 22:17:36

标签: java constructor initialization subclass inheritance

我有

class A {

    int var;
    public A(int x) {
        var = x;
    }
}


class B extends A {
     int var2;

     public B(int x, int y) {
         super(...);
         var2 = y;
         x = f(y);
     }
 }

对于子类B,我需要计算在A的构造函数中使用的值x。如果我可以将super移到x=f(y)之下,那么我可以将结果传递给A(超级)的构造函数。但是super必须是B构造函数中的第一行。

有没有办法在第一次使用适当的值初始化A?如果A.var是最终的并且在施工后我无法返回并改变它会怎么样?

当然,我可以放super(f(y)),但我可以想象这会变得困难的情况。

4 个答案:

答案 0 :(得分:10)

假设var是私有的,你需要用构造函数设置值(这似乎是问题的关键,否则有很多简单的解决方案),我会使用类似工厂的静态方法来做到这一点。

class B extends A {
     int var2;

     public static B createB(int x, int y) {
         x = f(y);
         return new B(x, y);
     }

     public B(x, y) {
         super(x);
         this.var2 = y;
     }
 }
这样的事情。您别无选择,因为显式构造函数调用必须发生在包装构造函数的第一行。

答案 1 :(得分:2)

你可以这样做:

class B extends A {
    int var2;

    public B(int x, int y) {
        super(calculateX(y));
        var2 = y;
    }

    private static int calculateX(int y) {
        return y;
    }
}

在调用超类构造函数之前,只能调用静态方法。

答案 2 :(得分:0)

替代hvgotcodes,因为看起来你也想要设置A的变量......

class B extends A {
 int var2;

 public B(int x, int y) {
     super(x);
     var2 = y;
     x = f(y);
     super.var = ...;
 }
}

答案 3 :(得分:0)

另一种方法可能是将没有参数的受保护构造函数添加到依赖于延迟初始化的A:

class A {
    int var;

    public A(int x) {
        var = x;
    }

    /**
    * Just use this constructor if you initialize var
    */
    protected A(int x) {
    }
}

class B extends A {
    int var2;

    public B(int x, int y) {
        super();
        var2 = y;
        var = f(y);
    }
}

但这很难记录,对团队成员来说不是很明显,也不应该用于api。