Java枚举之EnumSet详解

 更新时间:2023年12月20日 10:07:05   作者:Brain_L  
这篇文章主要介绍了Java枚举之EnumSet详解,使用时进行与或运算,但是定义多了之后,会很乱、臃肿,编写容易出错,EnumSet可以实现类似的功能,且使用起来很简洁,需要的朋友可以参考下

EnumSet

当需要使用位运算时,可能你会如此定义

private final static int FIRST = 1;
private final static int SECOND = 1 << 1;
private final static int THIRD = 1 << 2;

使用时进行与或运算。但是定义多了之后,会很乱、臃肿,编写容易出错。EnumSet可以实现类似的功能,且使用起来很简洁。

示例

public enum EnumTest {
    FIRST,
    SECOND;
    public static void main(String[] args) {
        EnumSet<EnumTest> enumTests = EnumSet.allOf(EnumTest.class);
        //输出enumTests所有元素
        System.out.println(enumTests);//[FIRST, SECOND]
        //判断集中中是否存在
        System.out.println(enumTests.contains(EnumTest.SECOND));//true
    }
}

属性 

final Class<E> elementType;
final Enum<?>[] universe;

elementType为EnumSet存储的元素类型,示例中为EnumTest。universe为存储的所有枚举实例数组。

noneOf

public static <E extends Enum<E>> EnumSet<E> noneOf(Class<E> elementType) {
        Enum<?>[] universe = getUniverse(elementType);//获取枚举类型的所有实例
        if (universe == null)
            throw new ClassCastException(elementType + " not an enum");
        if (universe.length <= 64)//判断枚举实例的数目
            return new RegularEnumSet<>(elementType, universe);
        else
            return new JumboEnumSet<>(elementType, universe);
    }

首先获取枚举的所有实例,具体过程参见上一篇博文。RegularEnumSet和JumboEnumSet为EnumSet的两个实现类。

当枚举中定义的实例数不大于64时,生成RegularEnumSet,否则生成JumboEnumSet。

allOf

public static <E extends Enum<E>> EnumSet<E> allOf(Class<E> elementType) {
        EnumSet<E> result = noneOf(elementType);
        result.addAll();
        return result;
    }

在allOf先调用noneOf获得EnumSet(RegularEnumSet或JumboEnumSet),allAll是个抽象方法,看下RegularEnumSet中是如何实现的

//RegularEnumSet.java
private long elements = 0L;
void addAll() {
        if (universe.length != 0)
            elements = -1L >>> -universe.length;
}

elements代表了EnumSet中存储的所有枚举实例所代表的bit数。示例中存储了FIRST和SECOND,分别代表了1和2,所以allAll之后,elements为3。

of

public static <E extends Enum<E>> EnumSet<E> of(E e) {
        EnumSet<E> result = noneOf(e.getDeclaringClass());
        result.add(e);
        return result;
    }

of方法有好几个,参数不同罢了。这里指只说单元素的情况。getDeclaringClass获取e的枚举类型。这里有个问题,为什么不用getClass。在示例中,这两种返回的都是EnumTest。再看另一种情况

public enum EnumTest {
    FIRST {
        @Override
        void doSometing() {
        }
    },
    SECOND {
        @Override
        void doSometing() {
        }
    };
    abstract void doSometing();
    public static void main(String[] args) {
        System.out.println(EnumTest.SECOND.getClass());//EnumTest$2
        System.out.println(EnumTest.SECOND.getDeclaringClass());//EnumTest
    }
}

当枚举有方法实现时,此时相当于内部类,getClass和getDeclaringClass返回的结果就不一样了。

public final Class<E> getDeclaringClass() {
        Class<?> clazz = getClass();//获取自己的Class
        Class<?> zuper = clazz.getSuperclass();//获取父类的Class
        return (zuper == Enum.class) ? (Class<E>)clazz : (Class<E>)zuper;//两者不等时取父类
    }

说回来,获取类型之后,调用noneOf。调用add方法。

//RegularEnumSet.java
public boolean add(E e) {
        typeCheck(e);//校验e的类型是否和生成elementType一致
        long oldElements = elements;
        elements |= (1L << ((Enum<?>)e).ordinal());
        return elements != oldElements;
    }
 

如果此时e为EnumTest.SECOND,ordinal为1,那么elements与运算后则为2。

contains

contains为AbstractCollection中的方法,EnumSet并没有覆写,而是交给两个子类去实现的。

//RegularEnumSet.java
public boolean contains(Object e) {
        if (e == null)//判空
            return false;
        Class<?> eClass = e.getClass();//校验类型
        if (eClass != elementType && eClass.getSuperclass() != elementType)
            return false;
        //比特位运算
        return (elements & (1L << ((Enum<?>)e).ordinal())) != 0;
    }

contains最终算的就是枚举值对应的比特位是否被置位。

总结

枚举自带自增长的属性ordinal,可以和位运算完美的结合在一起。

EnumSet的根本就是利用了这个特性,将ordinal和位运算封装起来,用户只需要像使用Set一样使用它。

到此这篇关于Java枚举之EnumSet详解的文章就介绍到这了,更多相关Java枚举EnumSet内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • java线程中断 interrupt 和 LockSupport解析

    java线程中断 interrupt 和 LockSupport解析

    这篇文章主要为大家介绍了java线程中断 interrupt 和 LockSupport示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-02-02
  • springcloud feign 接口指定接口服务ip方式

    springcloud feign 接口指定接口服务ip方式

    这篇文章主要介绍了springcloud feign 接口指定接口服务ip方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • 浅谈Java泛型让声明方法返回子类型的方法

    浅谈Java泛型让声明方法返回子类型的方法

    下面小编就为大家带来一篇浅谈Java泛型让声明方法返回子类型的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-02-02
  • javaweb启动时启动socket服务端代码实现

    javaweb启动时启动socket服务端代码实现

    这篇文章主要介绍了javaweb启动时启动socket服务端代码实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-11-11
  • Springboot实现高吞吐量异步处理详解(适用于高并发场景)

    Springboot实现高吞吐量异步处理详解(适用于高并发场景)

    这篇文章主要介绍了Springboot实现高吞吐量异步处理详解(适用于高并发场景),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-11-11
  • SpringBoot实现导出复杂对象到Excel文件

    SpringBoot实现导出复杂对象到Excel文件

    这篇文章主要为大家详细介绍了如何使用Hutool和EasyExcel两种方式来实现在Spring Boot项目中导出复杂对象到Excel文件,需要的小伙伴可以参考下
    2025-03-03
  • java接口性能从20s优化到500ms示例详解

    java接口性能从20s优化到500ms示例详解

    这篇文章主要为大家介绍了java接口性能从20s优化到500ms的操作技巧示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • OpenCV实现普通阈值

    OpenCV实现普通阈值

    这篇文章主要为大家详细介绍了OpenCV实现普通阈值,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • SpringBoot整合Spring Batch示例代码

    SpringBoot整合Spring Batch示例代码

    这篇文章主要来和大家一起探讨一下SpringBoot如何整合Spring Batch,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-10-10
  • springboot集成websocket的四种方式小结

    springboot集成websocket的四种方式小结

    本文主要介绍了springboot集成websocket的四种方式小结,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-12-12

最新评论