三、String的hashCode()深刻分析
public int hashCode() { int h = hash; if (h == 0 && value.length > 0) { char val[] = value; for (int i = 0; i < value.length; i++) { h = 31 * h + val[i]; } hash = h; } return h; }
推导出的公式以下:
s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1]
举个例子推导计算一下:
假设 n=3 i=0 -> h = 31 * 0 + val[0] i=1 -> h = 31 * (31 * 0 + val[0]) + val[1] i=2 -> h = 31 * (31 * (31 * 0 + val[0]) + val[1]) + val[2] h = 31*31*31*0 + 31*31*val[0] + 31*val[1] + val[2] h = 31^(n-1)*val[0] + 31^(n-2)*val[1] + val[2]
3.一、为何使用31做为计算的因子呢?
- 选择质数做为乘子,会大大下降hash冲突的几率。质数的值越大,hash冲突率越低
- 31参与乘法运算,能够被 JVM 优化,
31 * i = (i << 5) - i
- 使用 101 计算 hash code 容易致使整型溢出,致使计算精度丢失