HashMap的hash()

为何要有HashMap的hash()方法,难道不能直接使用KV中K原有的hash值吗?在HashMap的put、get操做时为何不能直接使用K中原有的hash值。html

/**
     * Computes key.hashCode() and spreads (XORs) higher bits of hash * to lower. Because the table uses power-of-two masking, sets of * hashes that vary only in bits above the current mask will * always collide. (Among known examples are sets of Float keys * holding consecutive whole numbers in small tables.) So we * apply a transform that spreads the impact of higher bits * downward. There is a tradeoff between speed, utility, and * quality of bit-spreading. Because many common sets of hashes * are already reasonably distributed (so don't benefit from * spreading), and because we use trees to handle large sets of * collisions in bins, we just XOR some shifted bits in the * cheapest possible way to reduce systematic lossage, as well as * to incorporate impact of the highest bits that would otherwise * never be used in index calculations because of table bounds. */ static final int hash(Object key) { int h;  return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16); }

从上面的代码能够看到key的hash值的计算方法。key的hash值高16位不变,低16位与高16位异或做为key的最终hash值。(h >>> 16,表示无符号右移16位,高位补0,任何数跟0异或都是其自己,所以key的hash值高16位不变。) 
这里写图片描述 
为何要这么干呢? 
这个与HashMap中table下标的计算有关。java

n = table.length; index = (n-1) & hash;

由于,table的长度都是2的幂,所以index仅与hash值的低n位有关,hash值的高位都被与操做置为0了。 
假设table.length=2^4=16。 
这里写图片描述 
由上图能够看到,只有hash值的低4位参与了运算。 
这样作很容易产生碰撞。设计者权衡了speed, utility, and quality,将高16位与低16位异或来减小这种影响。设计者考虑到如今的hashCode分布的已经很不错了,并且当发生较大碰撞时也用树形存储下降了冲突。仅仅异或一下,既减小了系统的开销,也不会形成的由于高位没有参与下标的计算(table长度比较小时),从而引发的碰撞。git

HashMap#tableSizeFor()

源码:github

static final int MAXIMUM_CAPACITY = 1 << 30; /** * Returns a power of two size for the given target capacity. */ static final int tableSizeFor(int cap) { int n = cap - 1; n |= n >>> 1; n |= n >>> 2; n |= n >>> 4; n |= n >>> 8; n |= n >>> 16; return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1; }

这个方法被调用的地方:算法

public HashMap(int initialCapacity, float loadFactor) { /**省略此处代码**/ this.loadFactor = loadFactor; this.threshold = tableSizeFor(initialCapacity); }

由此能够看到,当在实例化HashMap实例时,若是给定了initialCapacity,因为HashMap的capacity都是2的幂,所以这个方法用于找到大于等于initialCapacity的最小的2的幂(initialCapacity若是就是2的幂,则返回的仍是这个数)。 
下面分析这个算法: 
首先,为何要对cap作减1操做。int n = cap - 1; 
这是为了防止,cap已是2的幂。若是cap已是2的幂, 又没有执行这个减1操做,则执行完后面的几条无符号右移操做以后,返回的capacity将是这个cap的2倍。若是不懂,要看完后面的几个无符号右移以后再回来看看。 
下面看看这几个无符号右移操做: 
若是n这时为0了(通过了cap-1以后),则通过后面的几回无符号右移依然是0,最后返回的capacity是1(最后有个n+1的操做)。 
这里只讨论n不等于0的状况。 
第一次右移markdown

n |= n >>> 1;

因为n不等于0,则n的二进制表示中总会有一bit为1,这时考虑最高位的1。经过无符号右移1位,则将最高位的1右移了1位,再作或操做,使得n的二进制表示中与最高位的1紧邻的右边一位也为1,如000011xxxxxx。 
第二次右移数据结构

n |= n >>> 2;

注意,这个n已经通过了n |= n >>> 1; 操做。假设此时n为000011xxxxxx ,则n无符号右移两位,会将最高位两个连续的1右移两位,而后再与原来的n作或操做,这样n的二进制表示的高位中会有4个连续的1。如00001111xxxxxx 。 
第三次右移app

n |= n >>> 4;

此次把已经有的高位中的连续的4个1,右移4位,再作或操做,这样n的二进制表示的高位中会有8个连续的1。如00001111 1111xxxxxx 。 
以此类推 
注意,容量最大也就是32bit的正数,所以最后n |= n >>> 16; ,最多也就32个1,可是这时已经大于了MAXIMUM_CAPACITY ,因此取值到MAXIMUM_CAPACITY 。 
举一个例子说明下吧。 
这里写图片描述ide

这个算法着实牛逼啊!this

注意,获得的这个capacity却被赋值给了threshold。

this.threshold = tableSizeFor(initialCapacity);

开始觉得这个是个Bug,感受应该这么写:

this.threshold = tableSizeFor(initialCapacity) * this.loadFactor;

这样才符合threshold的意思(当HashMap的size到达threshold这个阈值时会扩容)。 
可是,请注意,在构造方法中,并无对table这个成员变量进行初始化,table的初始化被推迟到了put方法中,在put方法中会对threshold从新计算,put方法的具体实现请看这篇博文。

参考资料

1.Java HashMap工做原理及实现 
2.Java7的HashMap初始化变化 
3.java HashMap

相关文章
相关标签/搜索