Использование вложенных типов перечислений в Java
у меня есть структура данных, которая включает вложенные перечисления, так что я мог бы сделать что-то вроде следующего:
Drink.COFFEE.getGroupName();
Drink.COFFEE.COLUMBIAN.getLabel();
и если бы были объявления метода:
someMethod(Drink type)
someOtherMethod(DrinkTypeInterface type)
тогда я мог бы сказать (соответственно):
someMethod(Drink.COFFEE)
someOtherMethod(Drink.COFFEE.COLUMBIAN)
вот что я придумал:
public enum Drink {
COFFEE("Coffee");
private String groupName;
private Drink(String groupName) {
this.groupName = groupName;
}
public enum Coffee implements DrinkTypeInterface {
COLUMBIAN("Columbian Blend"),
ETHIOPIAN("Ethiopian Blend");
private String label;
private Coffee(String label) {
this.label = label;
}
public String getLabel() {
return this.label;
}
}
String getGroupName() {
return this.groupName;
}
}
интерфейс:
public interface DrinkTypeInterface {
public String getLabel();
}
Я думаю, что я просто пытаюсь понять, что лучший способ сделать это на Java, или если мне нужно напишите кучу if-заявлений, чтобы разобраться с индивидуальным напитком.значения.)( Любой помочь?
4 ответов
Drink.COFFEE.getGroupName();
Drink.COFFEE.COLUMBIAN.getLabel();
во - первых, этот пример кода, который вы дали, несколько нарушает "закон Деметры", поскольку поле экземпляра COLUMBIAN используется только для извлечения метки. Кроме того, с этой структурой, COLUMBIAN до будьте примером перечисления кофе, но я не думаю, что это то, что вы действительно собираетесь здесь.
someMethod(Drink type)
someOtherMethod(DrinkTypeInterface type)
someMethod(Drink.COFFEE)
someOtherMethod(Drink.COFFEE.COLUMBIAN)
то, что я собираю из вашего образца, заключается в том, что вы хотите иметь перечисление, которое содержит "тип группы" того, что такое фактический напиток, и тогда каждый из них имеет индивидуальные значения для конкретного вида напитка. Ваш пример дает кофе, но чай должен работать так же хорошо.
проблема в том, как вы разместили свои перечисления. Как я уже говорил, вам придется сделать COLUMBIAN примером перечисления кофе, но это не лучший способ структурировать это.
проблема в том, что у вас есть напиток, затем кофе/чай, а затем их отдельные типы. Но, если вы думаете об этом, хотя HerbalTea Чай, это также напиток - поэтому он не принадлежит как просто экземпляр чая.
но, если вы делаете напиток тип перечисление само по себе, вы получаете то, что хотели, и структура становится яснее. И благодаря интерфейсам и силе делегирования, как тип напитка, так и перечисление напитка могут быть обработаны таким же образом, как и в следующей программе примера:
public final class DrinkEnumExample {
public interface DrinkTypeInterface {
String getDisplayableType();
}
public static enum DrinkType implements DrinkTypeInterface {
COFFEE("Coffee"), TEA("Tea");
private final String type;
private DrinkType(final String type) {
this.type = type;
}
public String getDisplayableType() {
return type;
}
}
public static enum Drink implements DrinkTypeInterface {
COLUMBIAN("Columbian Blend", DrinkType.COFFEE),
ETHIOPIAN("Ethiopian Blend", DrinkType.COFFEE),
MINT_TEA("Mint", DrinkType.TEA),
HERBAL_TEA("Herbal", DrinkType.TEA),
EARL_GREY("Earl Grey", DrinkType.TEA);
private final String label;
private final DrinkType type;
private Drink(String label, DrinkType type) {
this.label = label;
this.type = type;
}
public String getDisplayableType() {
return type.getDisplayableType();
}
public String getLabel() {
return label;
}
}
public DrinkEnumExample() {
super();
}
public static void main(String[] args) {
System.out.println("All drink types");
for (DrinkType type : DrinkType.values()) {
displayType(type);
System.out.println();
}
System.out.println("All drinks");
for (Drink drink : Drink.values()) {
displayDrink(drink);
System.out.println();
}
}
private static void displayDrink(Drink drink) {
displayType(drink);
System.out.print(" - ");
System.out.print(drink.getLabel());
}
private static void displayType(DrinkTypeInterface displayable) {
System.out.print(displayable.getDisplayableType());
}
}
вывод этой программы выглядит следующим образом:
All drink types
Coffee
Tea
All drinks
Coffee - Columbian Blend
Coffee - Ethiopian Blend
Tea - Mint
Tea - Herbal
Tea - Earl Grey
теперь затем, если по какой-то причине вы не хотели, чтобы все ваши напитки были в одном перечислении, тогда я не понимал, что вы собираетесь делать. В этом случае, если у вас есть функциональность, которая охватывает перечисления, сделайте отдельные перечисления кофе и чая (и что угодно) и примените интерфейс к обоим (или более) перечислениям. Но, я думаю, вы пытались сгруппировать их вот так.
рассмотрите возможность использования EnumSet
для сбора различных типов Drink
, как предложил здесь.
Addendum: в качестве конкретного примера приведенный ниже код выдает показанный результат.
Coffee: Columbian Blend Coffee: Ethiopian Blend
код:
public static enum DrinkType {
COFFEE("Coffee"), TEA("Tea");
private final String displayName;
private DrinkType(final String displayName) {
this.displayName = displayName;
}
public String getDisplayName() {
return displayName;
}
}
public enum Drink {
COLUMBIAN(DrinkType.COFFEE, "Columbian Blend"),
ETHIOPIAN(DrinkType.COFFEE, "Ethiopian Blend"),
MINT_TEA(DrinkType.TEA, "Mint"),
HERBAL_TEA(DrinkType.TEA, "Herbal"),
EARL_GREY(DrinkType.TEA, "Earl Grey");
public static Set<Drink> coffees = EnumSet.of(COLUMBIAN, ETHIOPIAN);
public static Set<Drink> teas = EnumSet.range(MINT_TEA, EARL_GREY);
private String groupName;
private String drinkName;
private Drink(DrinkType type, String drinkName) {
this.groupName = type.getDisplayName();
this.drinkName = drinkName;
}
public String getGroupName() {
return this.groupName;
}
public String getDrinkName() {
return drinkName;
}
}
public static void main(String... args) {
for (Drink d : Drink.coffees) {
System.out.println(d.getGroupName() + ": " + d.getDrinkName());
}
}
вы можете сделать что-то вроде:
enum dogs {
boxer, collie;
}
enum cats {
siamese, tom
}
enum Animal {
cat(cats.tom), dog(dogs.boxer);
Animal(Enum e) {
this.e = e;
}
Object[] subValues() {
return e.getDeclaringClass().getEnumConstants();
}
final Enum e;
}
public class Main {
public static void main(String[] args) {
for (Animal animal : Animal.values()) {
System.out.print(animal);
for (Object o : animal.subValues())
System.out.print(" " + o);
System.out.println();
}
}
}
мне самому недавно было любопытно, можно ли это сделать несколько удовлетворительно. Это решение, с которым я закончил, чей API, я думаю, также ближе соответствует древовидной структуре перечислений, которую изначально хотел asker:
public interface Drink {
String groupName();
String label();
enum Coffee implements Drink {
COLUMBIAN("Columbian Blend"),
ETHIOPIAN("Ethiopian Blend");
private final String label;
Coffee(String label) {
this.label = label;
}
@Override
public String groupName() {
return "Coffee";
}
@Override
public String label() {
return label;
}
}
enum Tea implements Drink {
MINT("Mint"),
HERBAL("Herbal"),
EARL_GREY("Earl Grey");
private final String label;
Tea(String label) {
this.label = label;
}
@Override
public String groupName() {
return "Tea";
}
@Override
public String label() {
return label;
}
}
}
public static void main(String[] args) {
Drink choice = Drink.Tea.EARL_GREY;
System.out.println(choice.groupName()); // Tea
System.out.println(choice.label()); // Earl Grey
}