EnumMap & EnumSet

一、EnumMap

一)基本用法

    public static Map<Size, Integer> countBySize(List<Clothes> clothes) {
EnumMap<Size, Integer> map = new EnumMap<>(Size.class);
for (Clothes c : clothes) {
Size size = c.getSize();
Integer count = map.get(size);
if (count != null) {
map.put(size, count + 1);
} else {
map.put(size, 1);
}
}
return map;
} public static void main(String[] args) {
List<Clothes> clothes = Arrays.asList(new Clothes[]{
new Clothes("C001",Size.SMALL), new Clothes("C002", Size.LARGE),
new Clothes("C003", Size.LARGE), new Clothes("C004", Size.MEDIUM),
new Clothes("C005", Size.SMALL), new Clothes("C006", Size.SMALL),
});
System.out.println(countBySize(clothes)); //{SMALL=3, MEDIUM=1, LARGE=2}
}

EnumMap是保证顺序的:输出是按照键在枚举中的顺序。

二)实现原理

EnumMap的实例变量:

private final Class<K> keyType; //类型信息
private transient K[] keyUniverse; //表示键,是所有可能的枚举值
private transient Object[] vals; //表示键对应的值
private transient int size = 0; //表示键值对个数

基本构造方法:

public EnumMap(Class<K> keyType) {
this.keyType = keyType;
keyUniverse = getKeyUniverse(keyType); //最终调用了枚举类型的values方法,该方法返回所有可能的枚举值
vals = new Object[keyUniverse.length];
}

保存键值对的put方法:

public V put(K key, V value) {
//检查键的类型
typeCheck(key);
//获取索引
int index = key.ordinal();
//放入值value
Object oldValue = vals[index];
//如果有值为null打包null值,这是为了区别null值与没有值
vals[index] = maskNull(value);
if(oldValue == null)
size++;
return unmaskNull(oldValue);
}
    private static final Object NULL = new Object() {
public int hashCode() {
return 0;
}
public String toString() {
return "java.util.EnumMap.NULL";
}
};
private Object maskNull(Object value) {
return (value == null ? NULL : value);
}
private V unmaskNull(Object value) {
return(V) (value == NULL ? null : value);
}

二、EnumSet

EnumSet的实现与EnumMap没有任何关系,而是用极为精简高效的位向量实现的。

位向量是计算机程序中解决问题的一种常用方式,我们有必要理解和掌握。

一)基本用法

通过EnumSet的静态工厂方法,可以创建EnumSet对象,比如:

//创建一个指定类型的EnumSet,不含任何元素,创建的EnumSet实际类型是EnumSet的子类
public static <E extends Enum<E>> EnumSet<E> noneOf(Class<E> elementType)

用法举例:

public enum Day {
MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, SUNDAY
}
EnumSet<Day> days = EnumSet.noneOf(Day.class);
days.add(Day.FRIDAY);
days.add(Day.SUNDAY);
days.add(Day.MONDAY);
System.out.println(days); //[MONDAY, FRIDAY, SUNDAY]

EnumSet的其他静态工厂方法:

//初始集合包括指定枚举类型的所有枚举值
<E extends Enum<E>> EnumSet<E> allOf(Class<E> elementType)
//初始集合包括枚举值中指定范围的元素
<E extends Enum<E>> EnumSet<E> range(E from, E to)
//包含指定集合的补集
<E extends Enum<E>> EnumSet<E> complementOf(EnumSet<E> s)
//包含参数中的所有元素
<E extends Enum<E>> EnumSet<E> of(E e)
<E extends Enum<E>> EnumSet<E> of(E e1, E e2)
<E extends Enum<E>> EnumSet<E> of(E e1, E e2, E e3)
<E extends Enum<E>> EnumSet<E> of(E e1, E e2, E e3, E e4)
<E extends Enum<E>> EnumSet<E> of(E e1, E e2, E e3, E e4, E e5)
<E extends Enum<E>> EnumSet<E> of(E first, E... rest)
//包含参数中的所有元素
<E extends Enum<E>> EnumSet<E> copyOf(EnumSet<E> s)
<E extends Enum<E>> EnumSet<E> copyOf(Collection<E> c)

二、应用场景

Worker[] workers = new Worker[]{
new Worker("Tom", EnumSet.of(Day.FRIDAY, Day.MONDAY, Day.SATURDAY)),
new Worker("Jerry", EnumSet.of(Day.MONDAY, Day.WEDNESDAY, Day.THURSDAY)),
new Worker("Teddy", EnumSet.of(Day.TUESDAY, Day.FRIDAY, Day.WEDNESDAY))
};
//哪些天一个人都不会来
EnumSet<Day> allDays = EnumSet.allOf(Day.class);
for (Worker w : workers) {
allDays.remove(w.getWorkingDays());
}

三、实现原理

位向量:用一个位表示一个元素状态,用一组位表示集合状态,

每个位对应一个元素,而状态只可能有两种。

比如之前的枚举类Day,它有7个枚举值,一个Day的集合就可以用一个字节byte表示,

最高位不用,设为0,从右到左,每位对应一个枚举值,1表示包含该元素,0表示不包含该元素。

EnumSet的实现,首先是主要实例变量:

final Class<E> elementType; //表示类型信息
final Enum[] universe; //表示所有枚举值

EnumSet自身没有记录元素个数的变量,也没有位向量,它们是子类维护的。

对于RegularEnumSet,它用一个long类型表示位向量,代码为:

private long elements = 0L;

RegularEnumSet没有定义元素个数的变量,都是实时计算出来的:

public int size() {
return Long.bitCount(elements);
}

而对于JumboEnumSet,用一个long数组表示,有单独的size变量

private long elements[];
private int size = 0;

EnumSet的静态工厂方法:

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);
}

RegularEnum 和 SetJumboEnumSet的构造方法:

RegularEnumSet(Class<E>elementType, Enum[] universe) {
super(elementType, universe);
}
JumboEnumSet(Class<E>elementType, Enum[] universe) {
super(elementType, universe);
elements = new long[(universe.length + 63) >>> 6];
}
EnumSet(Class<E>elementType, Enum[] universe) {
this.elementType = elementType;
this.universe = universe;
}

其他源码略。

四、实现原理

对于只有两种状态,且需要进行集合运算的数据,使用位向量表示

、位运算进行处理,是计算机程序中一种常见的思维方式。

Java笔记(九)EnumMap & EnumSet的更多相关文章

  1. Java笔记(九)……面向对象I

    面向对象概念 理解面向对象 面向对象是相对面向过程而言 面向对象和面向过程都是一种思想 面向过程 强调的是功能行为 面向对象 将功能封装进对象,强调具备了功能的对象. 面向对象是基于面向过程的. 面向 ...

  2. Elasticsearch笔记九之优化

    Elasticsearch笔记九之优化 ).get(); } curl命令可以在linux中建立一个定时任务每天执行一次,同样java代码也可以建立一个定时器来执行. 2:内存设置之前介绍过es集群有 ...

  3. 多线程学习笔记九之ThreadLocal

    目录 多线程学习笔记九之ThreadLocal 简介 类结构 源码分析 ThreadLocalMap set(T value) get() remove() 为什么ThreadLocalMap的键是W ...

  4. 《Java笔记——基础语法》

    Java笔记--基础语法       一.字符串的拼接: 例如: System.out.println(""+"");     二.换行语句: 例如: Syst ...

  5. Effective Java笔记一 创建和销毁对象

    Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...

  6. 《MFC游戏开发》笔记九 游戏中的碰撞判定初步&怪物运动简单AI

    本系列文章由七十一雾央编写,转载请注明出处. http://blog.csdn.net/u011371356/article/details/9374935 作者:七十一雾央 新浪微博:http:// ...

  7. java笔记00-目录

    --2013年7月26日17:49:59 学习java已久,趁最近有空,写一个总结: java笔记01-反射:

  8. Java笔记:Java集合概述和Set集合

    本文主要是Java集合的概述和Set集合 1.Java集合概述 1)数组可以保存多个对象,但数组长度不可变,一旦在初始化数组时指定了数组长度,这个数组长度就是不可变的,如果需要保存数量变化的数据,数组 ...

  9. java笔记整理

    Java 笔记整理 包含内容     Unix Java 基础, 数据库(Oracle jdbc Hibernate pl/sql), web, JSP, Struts, Ajax Spring, E ...

随机推荐

  1. Nginx详解一:Nginx基础篇之环境准备

    环境确认: 1.确认系统网络可用 2.确认yum源可用 3.确认关闭iptabkes规则 查看是否有iptabkes规则:iptables -L 如果有的话:iptables -F关闭 保险起见也看看 ...

  2. 多线程相关-ThreadPoolExecutor

    应用层面: ThreadPoolExecutor: 创建多线程池执行器:new ThreadPoolExecutor(),创建方法最终都是走的以下这个构造方法: /** * Creates a new ...

  3. Java 写一段字符到指定的文本文档中,如果该文本文档不存在,则创建该文本文档

    写一段字符到指定的文本文档中,如果该文本文档不存在,则创建该文本文档 import java.io.File; import java.io.FileNotFoundException; import ...

  4. 开始接触python

    1.什么是语言? 语言是一个事物与另一个事物交流的介质 python是人与计算机交流的介质 能够被计算机所识别的表达方式即是编程语言 2.什么是编程? 编程就是程序员将想让计算机做的事情用编程语言表达 ...

  5. CentOS安装jdk的三种方法

    方法一:手动解压JDK的压缩包,然后设置环境变量 方法二:用yum安装JDK,(全称为 Yellow dog Updater, Modified)是一个在Fedora和RedHat以及CentOS中的 ...

  6. HDU 1573 X问题(中国剩余定理标准解法)

    X问题 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submis ...

  7. UE4 UPROPERTY UFUNCTION

    http://blog.csdn.net/sinat_27456831/article/details/52800514

  8. [转] Form 表单数据处理 简单教程 formidable 使用心得

    入门,高手见笑 表单数据一种是get方式, 另一种是post 方式 1.get方式 对于get方式,node处理起来非常简单 如以下代码: var urlParsed = url.parse(requ ...

  9. delphi使用sqlite数据库时的中文路径问题

    https://blog.csdn.net/yuehaiyang/article/details/4184198 如果数据库所在的路径是中文路径的话,根本运行不起来,会报错,因为sqlite用的是ut ...

  10. CTAP: Complementary Temporal Action Proposal Generation (ECCV2018)

    互补时域动作提名生成 这里的互补是指actionness score grouping 和 sliding window ranking这两种方法提proposal的结合,这两种方法各有利弊,形成互补 ...