private static class Holder {
/**
* Table capacity above which to switch to use alternative hashing.
*/
static final int ALTERNATIVE_HASHING_THRESHOLD;
static {
String altThreshold = java.security.AccessController.doPrivileged(
new sun.security.action.GetPropertyAction(
"jdk.map.althashing.threshold"));
int threshold;
try {
threshold = (null != altThreshold)
? Integer.parseInt(altThreshold)
: ALTERNATIVE_HASHING_THRESHOLD_DEFAULT;
// disable alternative hashing if -1
if (threshold == -1) {
threshold = Integer.MAX_VALUE;
}
if (threshold < 0) {
throw new IllegalArgumentException("value must be positive integer.");
}
} catch(IllegalArgumentException failed) {
throw new Error("Illegal value for 'jdk.map.althashing.threshold'", failed);
}
ALTERNATIVE_HASHING_THRESHOLD = threshold;
}
}
内部结构
从上图可以看出,HashMap是基于数组+链表的方式实现的。来看Entry这个内部类:
Entry,4个属性(key,value,next节点,hash值)
staticclassEntry<K,V> implementsMap.Entry<K,V> {
final K key;
V value;
Entry<K,V> next;
int hash;
/**
* Creates new entry.
*/
Entry(int h, K k, V v, Entry<K,V> n) {
value = v;
next = n;
key = k;
hash = h;
}
publicfinal K getKey(){
return key;
}
publicfinal V getValue(){
return value;
}
publicfinal V setValue(V newValue){
V oldValue = value;
value = newValue;
return oldValue;
}
publicfinalbooleanequals(Object o){
if (!(o instanceof Map.Entry))
returnfalse;
Map.Entry e = (Map.Entry)o;
Object k1 = getKey();
Object k2 = e.getKey();
if (k1 == k2 || (k1 != null && k1.equals(k2))) {
Object v1 = getValue();
Object v2 = e.getValue();
if (v1 == v2 || (v1 != null && v1.equals(v2)))
returntrue;
}
returnfalse;
}
publicfinalinthashCode(){
return (key==null ? 0 : key.hashCode()) ^
(value==null ? 0 : value.hashCode());
}
publicfinal String toString(){
return getKey() + "=" + getValue();
}
/**
* This method is invoked whenever the value in an entry is
* overwritten by an invocation of put(k,v) for a key k that's already
* in the HashMap.
*/voidrecordAccess(HashMap<K,V> m){
}
/**
* This method is invoked whenever the entry is
* removed from the table.
*/voidrecordRemoval(HashMap<K,V> m){
}
}
staticintindexFor(int h, int length){
// assert Integer.bitCount(length) == 1 : "length must be a non-zero power of 2";return h & (length-1);
}
put方法
public V put(K key, V value){
if (table == EMPTY_TABLE) {
//如果表没有初始化,则以阈值threshold的容量初始化表
inflateTable(threshold);
}
if (key == null)
//如果key值为null,调用putForNullKey方法,所以hashmap可以插入key和value为null的值return putForNullKey(value);
//计算key的hash值int hash = hash(key);
//计算hash值对应表的位置,即表下标int i = indexFor(hash, table.length);
for (Entry<K,V> e = table[i]; e != null; e = e.next) {
Object k;
if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
//如果hash值相等并且(key值相等或者key的equals方法相等),//则覆盖,返回旧的value
V oldValue = e.value;
e.value = value;
e.recordAccess(this);
return oldValue;
}
}
//修改字数+1
modCount++;
//如果没找到key没找到,则插入
addEntry(hash, key, value, i);
returnnull;
}
初始化表方法,
表容量必须是2的倍数
(roundUpToPowerOf2)
privatevoidinflateTable(int toSize){
// Find a power of 2 >= toSizeint capacity = roundUpToPowerOf2(toSize);
threshold = (int) Math.min(capacity * loadFactor, MAXIMUM_CAPACITY + 1);
table = new Entry[capacity];
initHashSeedAsNeeded(capacity);
}
获取大于或等于给定值的最小的2的倍数
privatestaticintroundUpToPowerOf2(int
number){
// assert number >= 0 : "number must be non-negative";return number >= MAXIMUM_CAPACITY
? MAXIMUM_CAPACITY
: (number > 1) ? Integer.highestOneBit((number - 1) << 1) : 1;
}
highestOneBit:返回小于给定值的最大的2的倍数
publicstaticinthighestOneBit(int i){
// HD, Figure 3-1
i |= (i >> 1); //其余位不管,把最高位的1覆盖到第二位,使前2位都是1
i |= (i >> 2); //同样的,把第3、4位置1,使前4位都是1
i |= (i >> 4); //...
i |= (i >> 8); //...
i |= (i >> 16); //最高位以及低位都是1return i - (i >>> 1); //返回最高位为1,其余位全为0的值
}
voidcreateEntry(int hash, K key, V value, int bucketIndex){
Entry<K,V> e = table[bucketIndex];
table[bucketIndex] = new Entry<>(hash, key, value, e);
size++;
}
get方法
public V get(Object key){
if (key == null)
return getForNullKey(); //如果key为null,直接去table[0]中找
Entry<K,V> entry = getEntry(key);
returnnull == entry ? null : entry.getValue();
}
private V getForNullKey(){
if (size == 0) {
returnnull;
}
for (Entry<K,V> e = table[0]; e != null; e = e.next) {
if (e.key == null)
return e.value;
}
returnnull;
}
staticfinalinttableSizeFor(int cap){
int n = cap - 1; //先进行-1操作,当cap已经是2的倍数时,最后+1,返回该数本身
n |= n >>> 1; //右移1位,再进行或操作,然后赋值给n,使最高位的1的下一位也变成1
n |= n >>> 2; //右移2位,使最高2位的1右移覆盖后2位的值,即最高4位均为1
n |= n >>> 4; //右移4位...
n |= n >>> 8; //右移8位...
n |= n >>> 16; //右移16位...//如果cap<=0,返回1,如果>MAXIMUM_CAPACITY,返回MAXIMUM_CAPACITY,否则,最后的n+1操作返回大于等于cap的最小的2的倍数return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
}