给出this Dr Dobbs article,特别是Builder模式,我们如何处理将Builder子类化的情况?举一个精简版本的示例,其中我们想要子类来添加转基因标签,一个简单的实现是:

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;                                                                                             
    }                                                                                                                            
}

子类:

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;                                                                                                 
    }                                                                                                                            
}

现在,我们可以编写如下代码:

GMOFacts.Builder b = new GMOFacts.Builder();
b.GMO(true).calories(100);

但是,如果我们下错了订单,一切都会失败:

GMOFacts.Builder b = new GMOFacts.Builder();
b.calories(100).GMO(true);

问题当然是NutritionFacts.Builder返回NutritionFacts.Builder,而不是GMOFacts.Builder,那么我们如何解决这个问题,或者有更好的模式可以使用?

注意:this answer to a similar question提供了我上面的类;我的问题是关于确保构建器调用按正确顺序进行的问题.

推荐答案

您可以使用泛型来解决它.我想这个叫做"Curiously recurring generic patterns"

将基类生成器方法的返回类型设为泛型参数.

public class NutritionFacts {

    private final int calories;

    public static class Builder<T extends Builder<T>> {

        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;
    }
}

现在,使用派生类构建器作为泛型参数实例化基构建器.

public class GMOFacts extends NutritionFacts {

    private final boolean hasGMO;

    public static class Builder extends NutritionFacts.Builder<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;
    }
}

Java相关问答推荐

在Java Stream上调用collect方法出现意外结果

无法处理批处理侦听器中的反序列化异常

使用联接和分页的SpringBoot Spring数据JPA

Java 21虚拟线程执行器的性能比池化操作系统线程的执行器差?

获取字符串中带空格的数字和Java中的字符

在Java 17中使用两个十进制数字分析时间时出错,但在Java 8中成功

在Ubuntu 23.10上使用mp3创建JavaFX MediaPlayer时出错

将关闭拍卖的TimerService

如何配置空手道以使用FeignClient或RestTemplate代替ApacheHttpClient

FETCH类型设置为LAZY,但它仍会发送第二个请求

Quarkus:运行时出现EnumConstantNotPresentException

如何在IntelliJ IDEA的Build.sbt中添加外部JAR文件?

如何在ApacheHttpClient 5中为单个请求设置代理?

JavaFX复杂项目体系 struct

如何使用Java对随机生成的字母数字优惠券代码进行过期设置

如何设计包含已知和未知键值对映射的Java类?

Cucumber java-maven-示例表-未定义一步

如何调查进程列表中不可见的活跃 MySQL 事务?

Vaadin Flow:设置密码显示按钮属性

Xml Reader 将 BMP 外部的字符解析为代理项对,这会导致无效的 xml