java builder类,对Java Builder类进行子类化

Give this Dr Dobbs article, and the Builder Pattern in particular, how do we handle the case of subclassing a Builder? Taking a cut-down version of the example where we want to subclass to add GMO labelling, a naive implementation would be:

public class NutritionFacts {

private final int calories;

public static class Builder {

private int calories = 0;

public Builder() {}

public Builder calories(int val) { calories = val; return this; }

public NutritionFacts build() { return new NutritionFacts(this); }

}

protected NutritionFacts(Builder builder) {

calories = builder.calories;

}

}

Subclass:

public class GMOFacts extends NutritionFacts {

private final boolean hasGMO;

public static class Builder extends NutritionFacts.Builder {

private boolean hasGMO = false;

public Builder() {}

public Builder GMO(boolean val) { hasGMO = val; return this; }

public GMOFacts build() { return new GMOFacts(this); }

}

protected GMOFacts(Builder builder) {

super(builder);

hasGMO = builder.hasGMO;

}

}

Now, we can write code like this:

GMOFacts.Builder b = new GMOFacts.Builder();

b.GMO(true).calories(100);

But, if we get the order wrong, it all fails:

GMOFacts.Builder b = new GMOFacts.Builder();

b.calories(100).GMO(true);

The problem is of course that NutritionFacts.Builder returns a NutritionFacts.Builder, not a GMOFacts.Builder, so how do we solve this problem, or is there a better Pattern to use?

Note: this answer to a similar question offers up the classes I have above; my question is regarding the problem of ensuring the builder calls are in the correct order.

解决方案

You can solve it using generics. I think this is called the "Curiously recurring generic patterns"

Make the return type of the base class builder methods a generic argument.

public class NutritionFacts {

private final int calories;

public static class Builder {

private int calories = 0;

public Builder() {}

public T calories(int val) {

calories = val;

return (T) this;

}

public NutritionFacts build() { return new NutritionFacts(this); }

}

protected NutritionFacts(Builder builder) {

calories = builder.calories;

}

}

Now instantiate the base builder with the derived class builder as the generic argument.

public class GMOFacts extends NutritionFacts {

private final boolean hasGMO;

public static class Builder extends NutritionFacts.Builder {

private boolean hasGMO = false;

public Builder() {}

public Builder GMO(boolean val) {

hasGMO = val;

return this;

}

public GMOFacts build() { return new GMOFacts(this); }

}

protected GMOFacts(Builder builder) {

super(builder);

hasGMO = builder.hasGMO;

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值