我有以下 BeanValidation 代码可以正常工作,并允许验证一个 bean 注释为:

  @EnumValue(enumClass = MyTestEnum.class)
  private String field;

  public enum MyTestEnum {
    VAL1, VAL2;
  }

仅当字段值为“VAL1”或“VAL2”时才会被验证。
public class EnumNameValidator implements ConstraintValidator<EnumValue, String> {

  private Set<String> AVAILABLE_ENUM_NAMES;

  @Override
  public void initialize(EnumValue enumValue) {
    Class<? extends Enum<?>> enumSelected = enumValue.enumClass();
    Set<? extends Enum<?>> enumInstances = Sets.newHashSet(enumSelected.getEnumConstants());
    AVAILABLE_ENUM_NAMES = FluentIterable
            .from(enumInstances)
            .transform(PrimitiveGuavaFunctions.ENUM_TO_NAME)
            .toImmutableSet();
  }

  @Override
  public boolean isValid(String value, ConstraintValidatorContext context) {
    if ( value == null ) {
      return true;
    } else {
      return AVAILABLE_ENUM_NAMES.contains(value);
    }
  }

}

我不明白的是为什么我的第一次尝试失败了。使用以下代码代替上面的 enumSelected.getEnumConstants():
Set<? extends Enum<?>> enumInstances = EnumSet.allOf(enumSelected);

Intellij 12 没有突出显示任何错误,但编译器说:
java: method allOf in class java.util.EnumSet<E> cannot be applied to given types;
  required: java.lang.Class<E>
  found: java.lang.Class<capture#1 of ? extends java.lang.Enum<?>>
  reason: inferred type does not conform to declared bound(s)
    inferred: capture#1 of ? extends java.lang.Enum<?>
    bound(s): java.lang.Enum<capture#1 of ? extends java.lang.Enum<?>>

我不明白这个问题,我也有可以正常工作的代码:
  private static <T extends Enum<T> & EnumAlternativeName> T safeGetByAlternativeName(Class<T> enumClass, String alternativeName) {
    for ( T t : EnumSet.allOf(enumClass) ) {
      if ( t.getAlternativeName().equals(alternativeName) ) {
        return t;
      }
    }
    return null;
  }

最佳答案

我的猜测是,在 ? extends Enum<?> 中,两个 ? 可能不同,而 allOf 期望 T extends Enum<T>,其中两个 0x251134312 是相同的。

例如,考虑以下代码:

static enum MyEnum {}
static class EnumValue<T extends Enum<T>> {
    Class<T> enumClass;
    EnumValue(Class<T> enumClass) {
        this.enumClass = enumClass;
    }
    Class<T> enumClass() { return enumClass; }
}

这些行将编译:
EnumValue<?> enumValue = new EnumValue(MyEnum.class); // raw constructor
Set<? extends Enum<?>> enumInstances = EnumSet.allOf(enumValue.enumClass());

因为我们知道 T 中的两个 T 是相同的,但这不会:
EnumValue enumValue = new EnumValue(MyEnum.class);
Class<? extends Enum<?>> enumSelected = enumValue.enumClass();
Set<? extends Enum<?>> enumInstances = EnumSet.allOf(enumSelected);

因为您使用 enumValue.enumClass() 作为中间步骤丢失了信息。

关于java - 泛型和类<?扩展 Enum<?>>、EnumSet.allOf(class) 与 class.getEnumConstants(),我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/15366375/

10-13 04:40