Как преобразовать строковый результат перечисления с переопределенной toString() обратно в enum?

StackOverflow https://stackoverflow.com/questions/240122

Вопрос

Учитывая следующее java-перечисление:

public enum AgeRange {

   A18TO23 {
        public String toString() {        
            return "18 - 23";
        }
    },
   A24TO29 {
        public String toString() {        
            return "24 - 29";
        }
    },
   A30TO35 {
        public String toString() {        
            return "30 - 35";
        }
    },

}

Есть ли какой-либо способ преобразовать строковое значение "18 - 23" в соответствующее значение enum, т. е.Возрастной диапазон.ОТ 18 до 23 лет?

Спасибо!

Это было полезно?

Решение

Самый лучший и простой способ сделать это выглядит следующим образом:

public enum AgeRange {
    A18TO23 ("18-23"),
    A24TO29 ("24-29"),
    A30TO35("30-35");

    private String value;

    AgeRange(String value){
        this.value = value;
    }

    public String toString(){
        return value;
    }

    public static AgeRange getByValue(String value){
        for (final AgeRange element : EnumSet.allOf(AgeRange.class)) {
            if (element.toString().equals(value)) {
                return element;
            }
        }
        return null;
    }
}

Затем вам просто нужно вызвать getByValue() метод с помощью String внесите в него свой вклад.

Другие советы

Вы всегда можете создать сопоставление из строки в значение - сделайте это статически, чтобы вам нужно было сопоставить его только один раз, предполагая, что возвращаемая строка остается неизменной с течением времени.Насколько мне известно, там нет ничего встроенного.

Согласно пункту 30 effective java (2nd ed), это может быть (это намного быстрее, чем цикл)

public enum AgeRange {
       A18TO23("18-23"),
       A24TO29("24-29"),
       A30TO35("30-35");

       private final String value;

       AgeRange(String value){
          this.value = value;
       }

       @Override public String toString(){
           return value;
       }

       private static final Map<String, AgeRange> stringToEnum =
           new HashMap<String, AgeRange>();

       static {
           for (AgeRange r : values()) {
               stringToEnum.put(r.toString(), r);
           }
       }

       public static AgeRange getByValue(String value){
           return stringToEnum.get(value);
       }
}
for (AgeRange ar: EnumSet.allOf(AgeRange)) {
    if (ar.toString().equals(inString)) {
         myAnswer = ar;
         break;
    }
}

Или что-то в этом роде?Только что ввел, не прогонял через компилятор.Простите (прокомментируйте) опечатки...

Или используйте подобную логику, чтобы построить карту один раз.Избегайте итерации во время выполнения.Хорошая идея, Джон.

Класс переопределяет "toString()" - итак, чтобы выполнить обратную операцию, вам нужно переопределить valueOf() чтобы перевести выходные данные toString() вернемся к значениям перечисления.

public enum AgeRange {

   A18TO23 {
        public String toString() {        
                return "18 - 23";
        }
        public AgeRange valueOf (Class enumClass, String name) {
                return A18T023
        }
    },

    .
    .
    .
}

Покупатель остерегается - несокомпилированный и непроверенный...

Механизм для toString() и valueOf() является документированной частью API

Вы могли бы попробовать что-то вроде следующего?

static AgeRange fromString(String range) {
    for (AgeRange ageRange : values()) {
        if (range.equals(ageRange.toString())) {
            return ageRange;
        }
    }
    return null;   
}

Или, как предлагали другие, используя подход кэширования:

private static Map<String, AgeRange> map;

private static synchronized void registerAgeRange(AgeRange ageRange) {
    if (map == null) {
        map = new HashMap<String, AgeRange>();
    }
    map.put(ageRange.toString(), ageRange);
}

AgeRange() {
    registerAgeRange(this);
}

static AgeRange fromString(String range) {
    return map.get(range);
}
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top