Получить экземпляр enum из класса

мне трудно выразить точный вопрос словами, поэтому я просто приведу пример.

у меня есть два Enum типа:

enum Shape {
    CAT, DOG;
}

enum Color {
    BLUE, RED;
}

у меня есть метод:

public Object getInstance(String value, Class<?> type);

Я хотел бы использовать метод, как:

// someValue is probably "RED", and someEnumClass is probably Color.class
Color c = getInstance(someValue, someEnumClass);

у меня возникли проблемы с определением того, как именно реализовать getInstance(). Как только вы знаете точное Enum класс, который вы хотите создать, это просто:

Color.valueOf("RED");

но как это может быть выше линия будет выполнена с неизвестным Class? (Однако известно, что someEnumClass является наследником Enum.)

спасибо!

4 ответов


 public static <T extends Enum<T>> T getInstance(final String value, final Class<T> enumClass) {
     return Enum.valueOf(enumClass, value);
 }

и метод должен использоваться как:

final Shape shape = getInstance("CAT", Shape.class);

опять же, вы всегда можете использовать

final Shape shape = Shape.valueOf("CAT");

, который является сокращением для

Enum.valueOf(Shape.class, "CAT");

мы хотим получить Method объект, который отражает valueOf метод переданного Class принимает


Итак, вот код, использующий весеннюю проверку и отлично работает для меня. Полный код приведен ниже.

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

import javax.validation.Constraint;
import javax.validation.Payload;
import javax.validation.ReportAsSingleViolation;
import javax.validation.constraints.NotNull;

@Documented
@Constraint(validatedBy = EnumValidatorImpl.class)
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
@NotNull(message = "Value cannot be null")
@ReportAsSingleViolation
public @interface EnumValidator {

  Class<? extends Enum<?>> enumClazz();

  String message() default "Value is not valid";

  Class<?>[] groups() default {};

  Class<? extends Payload>[] payload() default {};

}

реализация вышеуказанного класса:

import java.util.ArrayList;
import java.util.List;

import javax.validation.ConstraintValidator;
import javax.validation.ConstraintValidatorContext;

public class EnumValidatorImpl implements ConstraintValidator<EnumValidator, String> {

  List<String> valueList = null;

  @Override
  public boolean isValid(String value, ConstraintValidatorContext context) {
    if(!valueList.contains(value.toUpperCase())) {
      return false;
    }
    return true;
  }

  @Override
  public void initialize(EnumValidator constraintAnnotation) {
    valueList = new ArrayList<String>();
    Class<? extends Enum<?>> enumClass = constraintAnnotation.enumClazz();

    @SuppressWarnings("rawtypes")
    Enum[] enumValArr = enumClass.getEnumConstants();

    for(@SuppressWarnings("rawtypes")
    Enum enumVal : enumValArr) {
      valueList.add(enumVal.toString());
    }

  }

}

ИСПОЛЬЗОВАНИЕ ВЫШЕУКАЗАННОЙ АННОТАЦИИ ОЧЕНЬ ПРОСТО

 @JsonProperty("lead_id")
  @EnumValidator( enumClazz=DefaultEnum.class,message="This error is coming from the enum class", groups = {Group1.class })
  private String leadId;

поскольку у вас есть представление о том, какой класс вы ищете, вы можете просто спросить перечисление, если оно знает, что вас интересует:

public enum MyColor
{
  RED  ("red", Color.RED),
  BLUE ("blue", Color.BLUE),
  TAUPE ("brownish", new COLOR(80,64,77));

  private final String _name;
  private final Color _color;

  MyColor(String name, Color color)
  {
    _name = name;
    _color = color;
  }

  public static Color parseColor(String colorName)
  {
    for (MyColor mc : MyColor.values())
    {
      if (mc._name.equalsIgnoreCase(colorName))
        return mc._color;
    }
    return null;
  }
}

тем не менее, подключение строк к нескольким перечислениям в поисках соответствия ставит под угрозу безопасность типа, которую вы получаете с перечислениями. Если вы сопоставите "красный" с обоими MyColor.RED и NuclearThreatWarningLevel.RED тогда вы можете, по крайней мере, оказаться в неправильном классе. В худшем случае вы можете оказаться в своем подземном бункере в течение 6 месяцев, ожидая, пока воздух очистится, когда все вы хотели было машину покрасить в красный цвет:)

было бы лучше перепроектировать эту область вашего кода, если это возможно, поэтому вам не нужно динамически преобразовывать строку в экземпляр одного из нескольких классов. Если вы расширите свой ответ, чтобы включить проблему, которую вы пытаетесь решить, возможно, у сообщества SO будут некоторые идеи.