02、枚举可用于 switch 语句
这个我在之前的一篇我去的文章中详细地说明过了,感兴趣的小伙伴可以点击链接跳转过去看一下。
switch (playerType) { case TENNIS: return "网球运动员费德勒"; case FOOTBALL: return "足球运动员C罗"; case BASKETBALL: return "篮球运动员詹姆斯"; case UNKNOWN: throw new IllegalArgumentException("未知"); default: throw new IllegalArgumentException( "运动员类型: " + playerType); }
03、枚举可以有构造方法
如果枚举中需要包含更多信息的话,可以为其添加一些字段,比如下面示例中的 name,此时需要为枚举添加一个带参的构造方法,这样就可以在定义枚举时添加对应的名称了。
public enum PlayerType { TENNIS("网球"), FOOTBALL("足球"), BASKETBALL("篮球"); private String name; PlayerType(String name) { this.name = name; } }
04、EnumSet
EnumSet 是一个专门针对枚举类型的 Set 接口的实现类,它是处理枚举类型数据的一把利器,非常高效(内部实现是位向量,我也搞不懂)。
因为 EnumSet 是一个抽象类,所以创建 EnumSet 时不能使用 new 关键字。不过,EnumSet 提供了很多有用的静态工厂方法:
下面的示例中使用 noneOf() 创建了一个空的 PlayerType 的 EnumSet;使用 allOf() 创建了一个包含所有 PlayerType 的 EnumSet。
public class EnumSetTest { public enum PlayerType { TENNIS, FOOTBALL, BASKETBALL } public static void main(String[] args) { EnumSet<PlayerType> enumSetNone = EnumSet.noneOf(PlayerType.class); System.out.println(enumSetNone); EnumSet<PlayerType> enumSetAll = EnumSet.allOf(PlayerType.class); System.out.println(enumSetAll); } }
程序输出结果如下所示:
[]
[TENNIS, FOOTBALL, BASKETBALL]
有了 EnumSet 后,就可以使用 Set 的一些方法了:
05、EnumMap
EnumMap 是一个专门针对枚举类型的 Map 接口的实现类,它可以将枚举常量作为键来使用。EnumMap 的效率比 HashMap 还要高,可以直接通过数组下标(枚举的 ordinal 值)访问到元素。
和 EnumSet 不同,EnumMap 不是一个抽象类,所以创建 EnumMap 时可以使用 new 关键字:
EnumMap<PlayerType, String> enumMap = new EnumMap<>(PlayerType.class);
有了 EnumMap 对象后就可以使用 Map 的一些方法了:
和 HashMap 的使用方法大致相同,来看下面的例子:
EnumMap<PlayerType, String> enumMap = new EnumMap<>(PlayerType.class); enumMap.put(PlayerType.BASKETBALL,"篮球运动员"); enumMap.put(PlayerType.FOOTBALL,"足球运动员"); enumMap.put(PlayerType.TENNIS,"网球运动员"); System.out.println(enumMap); System.out.println(enumMap.get(PlayerType.BASKETBALL)); System.out.println(enumMap.containsKey(PlayerType.BASKETBALL)); System.out.println(enumMap.remove(PlayerType.BASKETBALL)); 1
程序输出结果如下所示:
{TENNIS=网球运动员, FOOTBALL=足球运动员, BASKETBALL=篮球运动员}
篮球运动员
true
篮球运动员
06、单例
通常情况下,实现一个单例并非易事,不信,来看下面这段代码
public class Singleton { private volatile static Singleton singleton; private Singleton (){} public static Singleton getSingleton() { if (singleton == null) { synchronized (Singleton.class) { if (singleton == null) { singleton = new Singleton(); } } } return singleton; } }
但枚举的出现,让代码量减少到极致:
public enum EasySingleton{ INSTANCE; }
完事了,真的超级短,有没有?枚举默认实现了 Serializable 接口,因此 Java 虚拟机可以保证该类为单例,这与传统的实现方式不大相同。传统方式中,我们必须确保单例在反序列化期间不能创建任何新实例。
07、枚举可与数据库交互
我们可以配合 Mybatis 将数据库字段转换为枚举类型。现在假设有一个数据库字段 check_type 的类型如下:
`check_type` int(1) DEFAULT NULL COMMENT '检查类型(1:未通过、2:通过)',
它对应的枚举类型为 CheckType,代码如下:
public enum CheckType { NO_PASS(0, "未通过"), PASS(1, "通过"); private int key; private String text; private CheckType(int key, String text) { this.key = key; this.text = text; } public int getKey() { return key; } public String getText() { return text; } private static HashMap<Integer,CheckType> map = new HashMap<Integer,CheckType>(); static { for(CheckType d : CheckType.values()){ map.put(d.key, d); } } public static CheckType parse(Integer index) { if(map.containsKey(index)){ return map.get(index); } return null; } }
1)CheckType 添加了构造方法,还有两个字段,key 为 int 型,text 为 String 型。
2)CheckType 中有一个public static CheckType parse(Integer index)方法,可将一个 Integer 通过 key 的匹配转化为枚举类型。
那么现在,我们可以在 Mybatis 的配置文件中使用 typeHandler 将数据库字段转化为枚举类型。
<resultMap id="CheckLog" type="com.entity.CheckLog">
<id property="id" column="id"/>
<result property="checkType" column="check_type" typeHandler="com.CheckTypeHandler"></result>
</resultMap>
其中 checkType 字段对应的类如下:
public class CheckLog implements Serializable { private String id; private CheckType checkType; public String getId() { return id; } public void setId(String id) { this.id = id; } public CheckType getCheckType() { return checkType; } public void setCheckType(CheckType checkType) { this.checkType = checkType; } }
CheckTypeHandler 转换器的类源码如下:
public class CheckTypeHandler extends BaseTypeHandler<CheckType> { @Override public CheckType getNullableResult(ResultSet rs, String index) throws SQLException { return CheckType.parse(rs.getInt(index)); } @Override public CheckType getNullableResult(ResultSet rs, int index) throws SQLException { return CheckType.parse(rs.getInt(index)); } @Override public CheckType getNullableResult(CallableStatement cs, int index) throws SQLException { return CheckType.parse(cs.getInt(index)); } @Override public void setNonNullParameter(PreparedStatement ps, int index, CheckType val, JdbcType arg3) throws SQLException { ps.setInt(index, val.getKey()); } }
CheckTypeHandler 的核心功能就是调用 CheckType 枚举类的 parse() 方法对数据库字段进行转换。