com.alibaba.toolkit.util.collection.DefaultHashMap Maven / Gradle / Ivy
/*
* Copyright (c) 2002-2012 Alibaba Group Holding Limited.
* All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.alibaba.toolkit.util.collection;
import java.io.IOException;
import java.io.Serializable;
import java.util.AbstractCollection;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.Arrays;
import java.util.Collection;
import java.util.ConcurrentModificationException;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
/**
*
* Hash表的一个实现, 实现了Map
接口.
*
*
* 这个hash表的实现完全类似JDK的HashMap
, 但作了如下改变, 以便于子类派生, 并实现特殊功能:
*
*
* - 将部分类成员设置成protected和friendly
* - 增加了一些方法和事件
*
*
* 和JDK的HashMap
一样, 这个实现具有以下特性:
*
*
* - 支持值为
null
的key和value
* - 没有进行任何
synchronized
操作, 因而不是线程安全的. 但可以通过以下操作实现线程安全:
*
*
*
* Map m = Collections.synchronizedMap(new DefaultHashMap(...));
*
*
* - 不保证hash表中的entry的顺序
* - 以几乎衡定的性能存取hash表中的每个entry
* - 从hash表中取得的任何
Iterator
具有fail-fast特性: 当hash表的结构被改变时, 调用
* Iterator.remove
和Iterator.add
方法时, 会掷出
* ConcurrentModificationException
. 这样确保不会出现不确定的情况.
*
*
* @author Michael Zhou
* @version $Id: DefaultHashMap.java,v 1.1 2003/07/03 07:26:16 baobao Exp $
*/
public class DefaultHashMap extends AbstractMap implements Map, Cloneable, Serializable {
/*
* ==========================================================================
* ==
*/
/* 常量 */
/*
* ==========================================================================
* ==
*/
/** 默认的初始容量 - 2的整数次幂
. */
private static final int DEFAULT_INITIAL_CAPACITY = 16;
/** 最大容量 - 2的整数次幂
. */
private static final int MAXIMUM_CAPACITY = 1 << 30;
/** 默认的负载系数 */
private static final float DEFAULT_LOAD_FACTOR = 0.75f;
/*
* ==========================================================================
* ==
*/
/* 成员变量 */
/*
* ==========================================================================
* ==
*/
/** Hash表, 长度可变 - 但长度必须是2的整数次幂
. */
protected transient Entry[] table;
/** Hash表中的entry数. */
protected transient int size;
/**
* 阈值. 当hash表中的entry数超过它时, 自动扩容(resize
). 其值等于
* capacity×loadFactor
.
*
* @serial 自动序列化字段
*/
protected int threshold;
/**
* 负载系数.
*
* @serial 自动序列化字段
*/
protected final float loadFactor;
/**
* 当hash表发生"结构改变"的计数. 所谓"结构改变",
* 是指hash表中entry的数目发生改变或内部结构改变(如resize
).
* 这个字段是为了实现fail-fast.
*/
protected transient volatile int modCount;
/** key的集合视图. */
private transient Set keySet = null;
/** entry的集合视图. */
private transient Set entrySet = null;
/** value的集合视图. */
private transient Collection values = null;
/*
* ==========================================================================
* ==
*/
/* 构造函数 */
/*
* ==========================================================================
* ==
*/
/** 创建一个空的hash表. 使用指定的默认的初始容量(16)和默认的负载系数(0.75). */
public DefaultHashMap() {
this(DEFAULT_INITIAL_CAPACITY, DEFAULT_LOAD_FACTOR);
}
/**
* 创建一个空的hash表. 使用指定的初始阈值和默认的负载系数(0.75).
*
* @param initialCapacity 初始容量.
*/
public DefaultHashMap(int initialCapacity) {
this(initialCapacity, DEFAULT_LOAD_FACTOR);
}
/**
* 创建一个空的hash表. 使用指定的初始容量和负载系数.
*
* @param initialCapacity 初始容量
* @param loadFactor 负载系数.
*/
public DefaultHashMap(int initialCapacity, float loadFactor) {
if (initialCapacity < 0) {
throw new IllegalArgumentException("Illegal initial capacity: " + initialCapacity);
}
if (initialCapacity > MAXIMUM_CAPACITY) {
initialCapacity = MAXIMUM_CAPACITY;
}
if (loadFactor <= 0 || Float.isNaN(loadFactor)) {
throw new IllegalArgumentException("Illegal load factor: " + loadFactor);
}
// 确保初始容量为2的整数次幂.
int capacity = 1;
while (capacity < initialCapacity) {
capacity <<= 1;
}
this.loadFactor = loadFactor;
this.threshold = (int) (capacity * loadFactor);
this.table = new Entry[capacity];
onInit();
}
/**
* 复制指定Map
内容相同的HashMap
. 使用默认的负载系数(0.75).
*
* @param map 要复制的Map
*/
public DefaultHashMap(Map map) {
this(Math.max((int) (map.size() / DEFAULT_LOAD_FACTOR) + 1, DEFAULT_INITIAL_CAPACITY), DEFAULT_LOAD_FACTOR);
putAllForCreate(map);
}
/*
* ==========================================================================
* ==
*/
/* 实现Map接口的方法 */
/*
* ==========================================================================
* ==
*/
/**
* 返回hash表中entry的个数.
*
* @return hash表中的entry数.
*/
@Override
public int size() {
return size;
}
/**
* 判断是否为空的hash表.
*
* @return 如果为空(size() == 0
), 则返回true
.
*/
@Override
public boolean isEmpty() {
return size == 0;
}
/**
* 返回指定key对应的value. 如果hash表中没有value对应key, 则返回null
. 但是返回
* null
并不总是代表没有value对应指定的key, 也有可能是指 value值本身为null
* . 可以通过方法containsKey
来区分这两 种情况.
*
* @param key 指定key所对应的value将被返回.
* @return 指定key对应的value, 如果没有value对应此key, 则返回null
.
*/
@Override
public Object get(Object key) {
Entry entry = getEntry(key);
return entry == null ? null : entry.getValue();
}
/**
* 如果hash表中包含指定key的entry, 则返回true
.
*
* @param key 测试指定的key是否存在.
* @return 如果key对应的entry存在, 则返回true
.
*/
@Override
public boolean containsKey(Object key) {
Entry entry = getEntry(key);
return entry != null;
}
/**
* 将指定的value和key关联. 如果已经有value和此key相关联, 则取代之, 并 返回被取代的value.
*
* @param key 要关联的key
* @param value 要和key关联的value
* @return 如果已经存在和此key相关联的value, 则返回此value. 否则返回null
. 返回
* null
也可能是因为被取代的这个value值为null
.
*/
@Override
public Object put(Object key, Object value) {
Entry entry = getEntry(key);
if (entry != null) {
Object oldValue = entry.getValue();
entry.setValue(value);
entry.onAccess();
return oldValue;
} else {
modCount++;
// 如果表中的项数即将超过阈值, 则容量倍增.
if (size >= threshold) {
resize(table.length * 2);
}
addEntry(key, value);
return null;
}
}
/**
* 将Map
中的所有项都加入到当前的Map
中. 如果有相同的key, 则替换之.
*
* @param map 要加入的Map
*/
@Override
public void putAll(Map map) {
// 一次性扩容, 以便容纳性加入的entry.
int n = map.size();
if (n == 0) {
return;
}
if (n >= threshold) {
n = (int) (n / loadFactor + 1);
if (n > MAXIMUM_CAPACITY) {
n = MAXIMUM_CAPACITY;
}
int capacity = table.length;
while (capacity < n) {
capacity <<= 1;
}
resize(capacity);
}
for (Iterator i = map.entrySet().iterator(); i.hasNext(); ) {
Map.Entry entry = (Map.Entry) i.next();
put(entry.getKey(), entry.getValue());
}
}
/**
* 将指定key的entry从hash表中删除(如果该entry存在的话).
*
* @param key 要被删除的entry的key
* @return 被删除的entry的value. 如果entry不存在, 则返回null
. 但是返回
* null
并不总是代表没有value对应指定的key, 也有可能是指 value值本身为
* null
.
*/
@Override
public Object remove(Object key) {
Entry entry = removeEntryForKey(key);
return entry == null ? null : entry.getValue();
}
/** 清除hash表中的所有entry. */
@Override
public void clear() {
modCount++;
Arrays.fill(table, null);
size = 0;
}
/**
* 判断hash表中是否有一个或多个entry具有指定的value.
*
* @param value 要测试的value
* @return 如果有一个或多个entry具有指定的value, 则返回true
*/
@Override
public boolean containsValue(Object value) {
Entry[] tab = table;
for (Entry element : tab) {
for (Entry entry = element; entry != null; entry = entry.next) {
if (eq(value, entry.getValue())) {
return true;
}
}
}
return false;
}
/**
* 取得key的集合视图. 这个集合是以hash表为基础的, 任何hash表的改变, 都会反射到这个集合, 反之亦然. 该集合支持删除操作,
* 删除集合中的key就删除了hash表中的相应entry. 可以通过以下方法删除一个entry:
* Iterator.remove
, Set.remove
,
* removeAll
, retainAll
, 和clear
.
* 但集合不支持add
或addAll
操作.
*
* @return key的集合视图
*/
@Override
public Set keySet() {
Set ks = keySet;
return ks != null ? ks : (keySet = new KeySet());
}
/**
* 取得value的集合视图. 这个集合是以hash表为基础的, 任何hash表的改变, 都会反射到这个集合, 反之亦然. 该集合支持删除操作,
* 删除集合中的key就删除了hash表中的相应entry. 可以通过以下方法删除一个entry:
* Iterator.remove
, Collection.remove
,
* removeAll
, retainAll
, 和clear
.
* 但集合不支持add
或addAll
操作.
*
* @return value的集合视图
*/
@Override
public Collection values() {
Collection vs = values;
return vs != null ? vs : (values = new Values());
}
/**
* 取得entry的集合视图. 这个集合是以hash表为基础的, 任何hash表的改变, 都会反射到这个集合, 反之亦然. 该集合支持删除操作,
* 删除集合中的key就删除了hash表中的相应entry. 可以通过以下方法删除一个entry:
* Iterator.remove
, Set.remove
,
* removeAll
, retainAll
, 和clear
.
* 但集合不支持add
或addAll
操作.
*
* @return entry的集合视图
*/
@Override
public Set entrySet() {
Set es = entrySet;
return es != null ? es : (entrySet = new EntrySet());
}
/*
* ==========================================================================
* ==
*/
/* 内部类 */
/*
* ==========================================================================
* ==
*/
/** Map.Entry
的实现. */
protected static class Entry extends DefaultMapEntry {
/** key的hash值. */
protected final int hash;
/** 相同hash值的entry是以链表的方式存放的, 这个引用指向链表中的下一个entry. */
protected Entry next;
/**
* 创建一个新的entry.
*
* @param h key的hash值
* @param k entry的key
* @param v entry的value
* @param n 链表中的下一个entry
*/
protected Entry(int h, Object k, Object v, Entry n) {
super(k, v);
next = n;
hash = h;
}
/** 当put(key, value)
方法被调用时, 如果entry已经存在将被覆盖时, 此方法被调用. */
protected void onAccess() {
}
/** 当entry将被删除时, 此方法被调用. */
protected void onRemove() {
}
}
/** 遍历器. */
private abstract class HashIterator implements Iterator {
/** 当前entry. */
private Entry current;
/** 下一个要返回的entry. */
private Entry next;
/** 创建iterator时的修改计数. */
private int expectedModCount;
/** 当前位置索引. */
private int index;
/** 创建一个遍历器. */
protected HashIterator() {
expectedModCount = modCount;
Entry[] t = table;
int i = t.length;
Entry n = null;
if (size != 0) { // advance to first entry
while (i > 0 && (n = t[--i]) == null) {
;
}
}
next = n;
index = i;
}
/**
* 返回遍历器中是否还有下一个entry.
*
* @return 如果遍历器中还有下一个entry, 返回true
*/
public boolean hasNext() {
return next != null;
}
/** 删除一个当前entry. 执行前必须先执行next()
方法. */
public void remove() {
if (current == null) {
throw new IllegalStateException();
}
checkForComodification();
Object k = current.getKey();
current = null;
DefaultHashMap.this.removeEntryForKey(k);
expectedModCount = modCount;
}
/**
* 取得下一个entry.
*
* @return 下一个entry
*/
protected Entry nextEntry() {
checkForComodification();
Entry entry = next;
if (entry == null) {
throw new NoSuchElementException();
}
Entry n = entry.next;
Entry[] t = table;
int i = index;
while (n == null && i > 0) {
n = t[--i];
}
index = i;
next = n;
return current = entry;
}
/** 检查是否同时被修改. */
private void checkForComodification() {
if (modCount != expectedModCount) {
throw new ConcurrentModificationException();
}
}
}
/** 取得hash表的key的遍历器. */
private class KeyIterator extends HashIterator {
/**
* 取得下一个key.
*
* @return 下一个key
*/
public Object next() {
return nextEntry().getKey();
}
}
/** 取得hash表的value的遍历器. */
private class ValueIterator extends HashIterator {
/**
* 取得下一个value.
*
* @return 下一个value
*/
public Object next() {
return nextEntry().getValue();
}
}
/** 取得hash表的entry的遍历器. */
private class EntryIterator extends HashIterator {
/**
* 取得下一个entry.
*
* @return 下一个entry
*/
public Object next() {
return nextEntry();
}
}
/** key的集合视图. */
private class KeySet extends AbstractSet {
/**
* 取得key的遍历器.
*
* @return key的遍历器
*/
@Override
public Iterator iterator() {
return newKeyIterator();
}
/**
* 取得集合的大小, 就是hash表中entry的数量.
*
* @return hash表中entry的数量
*/
@Override
public int size() {
return size;
}
/**
* 判断key中是否包含指定对象.
*
* @param o 要查找的对象
* @return 如果key中包含指定的对象, 则返回true
*/
@Override
public boolean contains(Object o) {
return containsKey(o);
}
/**
* 从hash表中删除key为指定对象的entry.
*
* @param o 指定的key
* @return 如果删除成功, 则返回true
*/
@Override
public boolean remove(Object o) {
return DefaultHashMap.this.removeEntryForKey(o) != null;
}
/** 清除所有entry. */
@Override
public void clear() {
DefaultHashMap.this.clear();
}
}
/** value的集合视图. */
private class Values extends AbstractCollection {
/**
* 取得value的遍历器.
*
* @return value的遍历器
*/
@Override
public Iterator iterator() {
return newValueIterator();
}
/**
* 取得集合的大小, 就是hash表中entry的数量.
*
* @return hash表中entry的数量
*/
@Override
public int size() {
return size;
}
/**
* 判断value中是否包含指定对象.
*
* @param o 要查找的对象
* @return 如果value中包含指定的对象, 则返回true
*/
@Override
public boolean contains(Object o) {
return containsValue(o);
}
/** 清除所有entry. */
@Override
public void clear() {
DefaultHashMap.this.clear();
}
}
/** entry的集合视图. */
private class EntrySet extends AbstractSet {
/**
* 取得entry的遍历器.
*
* @return entry的遍历器
*/
@Override
public Iterator iterator() {
return newEntryIterator();
}
/**
* 判断entry集合中是否包含指定对象.
*
* @param o 要查找的对象
* @return 如果entry中是否包含指定对象, 则返回true
*/
@Override
public boolean contains(Object o) {
if (!(o instanceof Map.Entry)) {
return false;
}
Map.Entry entry = (Map.Entry) o;
Entry candidate = getEntry(entry.getKey());
return eq(candidate, entry);
}
/**
* 从hash表中删除指定entry.
*
* @param o 要删除的entry
* @return 如果删除成功, 则返回true
*/
@Override
public boolean remove(Object o) {
return removeEntry(o) != null;
}
/**
* 取得集合的大小, 就是hash表中entry的数量.
*
* @return hash表中entry的数量
*/
@Override
public int size() {
return size;
}
/** 清除所有entry. */
@Override
public void clear() {
DefaultHashMap.this.clear();
}
}
/*
* ==========================================================================
* ==
*/
/* 序列化 */
/*
* ==========================================================================
* ==
*/
/** 序列化版本号. */
private static final long serialVersionUID = 362498820763181265L;
/**
* 从输入流中重建hash表(也就是反序列化).
*
* @param is 输入流
* @throws IOException 输入流异常
* @throws ClassNotFoundException 类未找到
*/
private void readObject(java.io.ObjectInputStream is) throws IOException, ClassNotFoundException {
// 读入threshold, loadfactor, 和其它隐藏的对象.
is.defaultReadObject();
// 取得hash表的容量.
int numBuckets = is.readInt();
table = new Entry[numBuckets];
// 给子类一个机会初始化.
onInit();
// 读入hash表中entry的个数.
int size = is.readInt();
// 读入所有的entry.
for (int i = 0; i < size; i++) {
Object key = is.readObject();
Object value = is.readObject();
putForCreate(key, value);
}
}
/**
* 将hash表的状态保存到输出流中(也就是"序列化").
*
* @param os 输出流
* @throws IOException 输出流异常
*/
private void writeObject(java.io.ObjectOutputStream os) throws IOException {
// 输出threshold, loadfactor, 和其它隐藏的对象.
os.defaultWriteObject();
// 输出hash表的容量.
os.writeInt(table.length);
// 输出hash表的大小.
os.writeInt(size);
// 输出所有entry.
for (Iterator i = entrySet().iterator(); i.hasNext(); ) {
Map.Entry entry = (Map.Entry) i.next();
os.writeObject(entry.getKey());
os.writeObject(entry.getValue());
}
}
/*
* ==========================================================================
* ==
*/
/* 复制方法(Clonable接口) */
/*
* ==========================================================================
* ==
*/
/**
* "浅"拷贝hash表, key和value本身并不被复制.
*
* @return 被复制的hash表.
*/
@Override
public Object clone() {
DefaultHashMap result = null;
try {
result = (DefaultHashMap) super.clone();
} catch (CloneNotSupportedException e) {
throw new InternalError(); // 不支持clone(不可能).
}
result.table = new Entry[table.length];
result.entrySet = null;
result.modCount = 0;
result.size = 0;
result.onInit();
result.putAllForCreate(this);
return result;
}
/*
* ==========================================================================
* ==
*/
/* 内部方法 */
/*
* ==========================================================================
* ==
*/
/**
* 给子类一个机会初始化自己. 该方法被所有构造函数以及"伪构造函数"(clone
,
* readObject
)调用. 调用时, hash表已被初始化, 但数据尚未被插入到表中.
*/
protected void onInit() {
}
/**
* 返回指定key对应的entry. 如果不存在, 则返回null.
*
* @param key 返回指定key对应的entry
* @return 指定key对应的entry
*/
protected Entry getEntry(Object key) {
int hash = hash(key);
int i = indexFor(hash, table.length);
for (Entry entry = table[i]; entry != null; entry = entry.next) {
if (entry.hash == hash && eq(key, entry.getKey())) {
return entry;
}
}
return null;
}
/**
* 加入一个entry到hash表中, 但不会对hash表进行resize()
操作. 子类可以覆盖此方法, 以改变
* put
, new HashMap(Map)
, clone
, 和
* readObject
方法的行为.
*
* @param key hash表的key
* @param value hash表的value
*/
protected void addEntry(Object key, Object value) {
int hash = hash(key);
int i = indexFor(hash, table.length);
table[i] = new Entry(hash, key, value, table[i]);
size++;
}
/**
* 此方法被构造函数或"伪构造函数"(clone, readObject)调用, 功能同put方法,
* 但不会调用resize或改变modCount计数.
*
* @param key 要关联的key
* @param value 要和key关联的value
*/
private void putForCreate(Object key, Object value) {
Entry entry = getEntry(key);
if (entry != null) {
entry.setValue(value);
} else {
addEntry(key, value);
}
}
/**
* 一次put多个entry.
*
* @param map 指定map的所有entry都被放入hash表中
*/
private void putAllForCreate(Map map) {
for (Iterator i = map.entrySet().iterator(); i.hasNext(); ) {
Map.Entry entry = (Map.Entry) i.next();
putForCreate(entry.getKey(), entry.getValue());
}
}
/**
* 删除指定key对应的entry, 并返回被删除的entry.
*
* @param key 要删除的entry的key
* @return 被删除的entry, 如果entry不存在, 则返回null
*/
protected Entry removeEntryForKey(Object key) {
int hash = hash(key);
int i = indexFor(hash, table.length);
Entry prev = table[i];
Entry entry = prev;
while (entry != null) {
Entry next = entry.next;
if (entry.hash == hash && eq(key, entry.getKey())) {
modCount++;
size--;
if (prev == entry) {
table[i] = next;
} else {
prev.next = next;
}
entry.onRemove();
return entry;
}
prev = entry;
entry = next;
}
return entry;
}
/**
* 删除指定的entry. 这个方法用于EntrySet.remove
.
*
* @param o 要删除的entry
* @return 被删除的entry
*/
protected Entry removeEntry(Object o) {
if (!(o instanceof Map.Entry)) {
return null;
}
Map.Entry entry = (Map.Entry) o;
Object key = entry.getKey();
int hash = hash(key);
int i = indexFor(hash, table.length);
Entry prev = table[i];
Entry e = prev;
while (e != null) {
Entry next = e.next;
if (e.hash == hash && e.equals(entry)) {
modCount++;
size--;
if (prev == e) {
table[i] = next;
} else {
prev.next = next;
}
e.onRemove();
return e;
}
prev = e;
e = next;
}
return e;
}
/**
* 子类覆盖此方法, 用来创建key的遍历器.
*
* @return hash表的key的遍历器
*/
protected Iterator newKeyIterator() {
return new KeyIterator();
}
/**
* 子类覆盖此方法, 用来创建value的遍历器.
*
* @return hash表的key的遍历器
*/
protected Iterator newValueIterator() {
return new ValueIterator();
}
/**
* 子类覆盖此方法, 用来创建entry的遍历器.
*
* @return hash表的key的遍历器
*/
protected Iterator newEntryIterator() {
return new EntryIterator();
}
/**
* 返回对象的hash值.
*
* @param obj 取得指定对象的hash值
* @return 指定对象的hash值
*/
protected static int hash(Object obj) {
int h = obj == null ? 0 : obj.hashCode();
return h - (h << 7); // 也就是, -127 * h
}
/**
* 比较两个对象.
*
* @param x 第一个对象
* @param y 第二个对象
* @return 如果相同, 则返回true
*/
protected static boolean eq(Object x, Object y) {
return x == null ? y == null : x == y || x.equals(y);
}
/**
* 返回索引值, 根据指定的hash值和数组的长度.
*
* @param hash hash值
* @param length 数组的长度, 必然是2的整数次幂
* @return hash值在数组中的序号
*/
protected static int indexFor(int hash, int length) {
return hash & length - 1;
}
/**
* 对map进行扩容. 此方法在entry数超过阈值时被调用.
*
* @param newCapacity 新的容量(必须为2的整数次幂).
*/
protected void resize(int newCapacity) {
Entry[] oldTable = table;
int oldCapacity = oldTable.length;
if (size < threshold || oldCapacity > newCapacity) {
return;
}
Entry[] newTable = new Entry[newCapacity];
transfer(newTable);
table = newTable;
threshold = (int) (newCapacity * loadFactor);
}
/**
* 将所有entry从当前表中移到新表中(扩容).
*
* @param newTable 新表
*/
protected void transfer(Entry[] newTable) {
Entry[] src = table;
int newCapacity = newTable.length;
for (int j = 0; j < src.length; j++) {
Entry entry = src[j];
if (entry != null) {
src[j] = null;
do {
Entry next = entry.next;
int i = indexFor(entry.hash, newCapacity);
entry.next = newTable[i];
newTable[i] = entry;
entry = next;
} while (entry != null);
}
}
}
/**
* 取得hash表的容量.
*
* @return hash表的容量
*/
protected int getCapacity() {
return table.length;
}
/**
* 取得hash表的负载系数.
*
* @return hash表的负载系数
*/
protected float getLoadFactor() {
return loadFactor;
}
/**
* 取得hash表的阈值.
*
* @return hash表的阈值
*/
protected int getThreshold() {
return threshold;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy