我可以在Java Enum上使用构建器模式吗?

Pur*_*ret 16 java enums builder

我正在重写一些代码,并且我已经决定了重新创建类的方法,因为有固定数量的工作表,我将它们创建为枚举.这是基于构建器模式与伸缩构造器的可读性的决定.

代码我抓住了一些.xls文件,添加了标题(并从其他.xls文件中读取了一些文件),也许还有一些子表格.然后,它以特定的方式将各种这些工作表合并在一起,以在主要的Excel工作簿上制作选项卡.我的问题是,某些工作簿选项卡使用不同数量的工作表作为参数.我正在尝试应用构建器模式.这是我正在尝试编写的代码:

public enum workBookSheet {
    mySheet1("Name1","mainSheet1.xls",true,1).addSubSheet("pathToSubSheet1.xls"),
    mySheet2("Name2","mainSheet2.xls",true,2).addHeaderSheet("pathToHeaders.xls").addSubsheet("pathtoSubSheet2.xls");

    private String tabName;
    private String mainSheetName;
    private Boolean available;
    private Integer order;
    private String subSheetName;
    private String headerSheetName;

    private workBookSheet(String tabName, String mainSheetName, Boolean available, Integer order){
        this.tabName = tabName;
        this.mainSheetName = mainSheetName;
        this.available = available;
        this.order = order;
    }
    public workBookSheet addSubSheet(String subSheetName){
        this.subSheetName = subSheetName;
        return this;
    }
    public workBookSheet addHeaderSheet(String headerSheetName){
        this.headerSheetName = headerSheetName;
        return this;
    }

}
Run Code Online (Sandbox Code Playgroud)

java给我的错误似乎是说Java期望我的枚举声明(顶部的'枚举构造函数'的逗号分隔列表)只包含构造函数,而不是其他方法.我可以将这些方法移到下面的"构建器"方法中,无需投诉.

public void buildSheets(){
    mySheet1.addSubSheet("pathToSubSheet1.xls");
    mySheet2.addHeaderSheet("pathToHeaders.xls").addSubSheet("pathtoSubSheet2.xls");
}
Run Code Online (Sandbox Code Playgroud)

这是在枚举上实现构建器模式的唯一方法吗?它确实需要我运行一个单独的方法,这不是太麻烦.虽然我觉得我打破了这种模式(我想,如果这样做的话,这不是一件坏事.)

注意我已经仔细研究过,看看是否有其他人在网上或其他地方提出这个问题.我发现的最接近的是关于Enums和Factories的问题,但这并没有完全回答我的问题.另外我知道这不是构建器模式,因为我没有单独的类,然后接受创建新枚举的build()方法.我想这是我初始设计中问题的根源,但我对Java相对较新.

那么有没有更好的方法在Java枚举上使用构建器模式?或者我的'足够接近'是什么?

Eme*_*gia 23

虽然它并不严格符合建造者模式,但简短的答案是肯定的.有点.

缺少的部分无法调用.build()实例化枚举常量,因为build()无法使用new.但是你可以从构建器模式中获得很多好处.让我们面对它,你不能使用静态工厂方法,并且enum常量的内联子类化是很奇怪的.

这是使用Country枚举的示例.

package app;

import org.apache.commons.lang.StringUtils;
import javax.annotation.Nullable;
import java.util.EnumSet;
import java.util.Set;
import static app.Language.*;
import static com.google.common.base.Preconditions.*;

enum Language {
    ITALIAN,
    ENGLISH,
    MALTESE
}

public enum Country {

    ITALY(new Builder(1, "Italy").addLanguage(ITALIAN)),
    MALTA(new Builder(2, "Malta").addLanguages(MALTESE, ENGLISH, ITALIAN).setPopulation(450_000));

    final private int id;
    final private String name;
    final private Integer population;
    final private Set<Language> languages;

    private static class Builder {

        private int id;
        private String name;
        private Integer population;
        private Set<Language> languages = EnumSet.noneOf(Language.class);

        public Builder(int id, String name) {
            checkArgument(!StringUtils.isBlank(name));

            this.id = id;
            this.name = name;
        }

        public Builder setPopulation(int population) {
            checkArgument(population > 0);

            this.population = population;
            return this;
        }

        public Builder addLanguage(Language language) {
            checkNotNull(language);

            this.languages.add(language);
            return this;
        }

        public Builder addLanguages(Language... language) {
            checkNotNull(language);

            this.languages.addAll(languages);
            return this;
        }
    }

    private Country(Builder builder) {

        this.id = builder.id;
        this.name = builder.name;
        this.population = builder.population;
        this.languages = builder.languages;

        checkState(!this.languages.isEmpty());
    }

    public int getId() {
        return id;
    }

    public String getName() {
        return name;
    }

    @Nullable
    public Integer getPopulation() {
        return population;
    }

    public Set<Language> getLanguages() {
        return languages;
    }
}
Run Code Online (Sandbox Code Playgroud)

如果您有构建常量的常用方法,甚至可以在构建器中放置静态工厂方法.

所以它不是布洛赫的建造者,但它非常接近.

  • 两者兼而有之.我想使用一个建设者,找到一些答案,但不喜欢它们,并想出这样做.然后我想我会在这个问题上分享我的解决方案,以防有人帮忙. (3认同)

Boh*_*ian 6

您可以使用实例块(通常被错误地称为“双括号初始化器”)来使用任意代码自定义构造:

public enum workBookSheet {

    mySheet1("Name1", "mainSheet1.xls", true, 1) {{
        addSubSheet("pathToSubSheet1.xls");
    }},
    mySheet2("Name2", "mainSheet2.xls", true, 2) {{
        // you can use the fluent interface:
        addHeaderSheet("pathToHeaders.xls").addSubSheet("pathtoSubSheet2.xls");
        // but I would prefer coding separate statements:
        addHeaderSheet("pathToHeaders.xls");
        addSubSheet("pathtoSubSheet2.xls");
    }};

    // rest of your class the same...
}
Run Code Online (Sandbox Code Playgroud)

使用此语法可让您解决由 an 强加的限制,enum但仍具有 builder/fluent 模式的简洁性、便利性和灵活性。

  • setter 不能是私有的(奇怪的错误消息:`Error:(8, 14) java: non-static method id(java.lang.String) cannot be referenced from a static context`),但是对于受保护的方法,这是可行的. 谢谢 (3认同)