1. 前言
Java枚举在开发中是非常实用的。今天再来分析几个小技巧并且回答一些同学的的疑问。首先要说明的是我的枚举建立在以下的范式之中:
2. 如何把枚举值绑定的下拉列表
这种场景非常常见,如果你把状态、类别等属性封装成枚举的结构,就像下面一样,一个标识对应一个状态,这是典型的下拉列表结构。
public enum EnabledEnum implements Enumerator {
/**
* Disable status enum.
*/
DISABLE(0, "不可用"),
/**
* Enable status enum.
*/
ENABLE(1, "可用");
private final int value;
private final String description;
EnabledEnum(int value, String description) {
this.value = value;
this.description = description;
}
@Override
public int value() {
return this.value;
}
@Override
public String description() {
return this.description;
}
}
前端希望能够获取这些状态作为下拉列表的填充,你该如何解析封装它呢?这里我有两个办法,第一个你可以从JSON 类库 Jackson 优雅序列化 Java 枚举类一文获得解决方案;第二种更加简单一些,写一个工具类就可以了。
public static & Enumerator> Map enumToMap(Class clazz) {
E[] enumConstants = clazz.getEnumConstants();
Map map = new LinkedHashMap<>();for (E e : enumConstants) {
map.put(e.value(), e.description());
}return map;
}
这里用了Java 8的新特性附加约束。也就是说extends
后面可以在通过&
符号附加额外约束,可以重复使用,注意必须为接口类型,不能为抽象类或者其他Class
。表示泛型的上界受到多个约束的制约。 & Enumerator>
可以解读为E
必须为一个枚举类而且同时还必须实现Enumerator
接口。
为什么实现
Enumerator
接口?第一个好处是可以规定统一的范式;第二个好处是解析的时候不用转换了。
此外我尝试用Stream
来封装,但是发现过于复杂了,虽然成功了但是有点难以理解,这里也分享一下:
public static & Enumerator> Map enumToOptions(Class enumClazz){// 合并时检查 key 是否重复
BinaryOperator merge = (u, v) -> {throw new IllegalStateException(String.format("Duplicate key %s", u));
};
Enumerator[] enumConstants = enumClazz.getEnumConstants();return Stream.of(enumConstants)
.collect(Collectors.toMap(Enumerator::value,
Enumerator::description,
merge,LinkedHashMap::new));
}
3. 如何根据值找到枚举
这种也很常见,最直接的方式是写一个switch语句。但是每个类都写一个就非常繁琐。所以这个也可以写一个工具类,这次用Stream
就简单的多了。
public static & Enumerator> Optional getEnumByValue(Class enumClazz, final Integer value){return Stream.of(enumClazz.getEnumConstants())
.filter(enumerator -> Objects.equals(enumerator.code(),value))
.findAny();
}
返回
Optional
是因为有可能给出的value
没有对应的枚举。
4. 总结
今天分享了两个小工具类来操作枚举,不光运用了枚举的一些知识,同时也使用了Java 8的三个新特性:泛型附加约束、Optional和Stream API。这些知识点以往我都分享过,有兴趣的可以到 felord.cn 搜索一下。如果你有更好的办法可以留言讨论。多多关注:码农小胖哥,更多原创干货分享。