HashMap下标计算详解

2022-06-09  本文已影响0人  我是光芒万丈

HashMap 计算方法为:(n - 1) & hash 具体见putVal
解读:由于n-1高位全部为0 因此(n - 1) & hash只会得到一个小于等于n-1的值,即在桶长度取值范围内.
为何不用取余运算,写一段模拟程序,我们来对比下速度:

        Random random = new Random();
        int[] array =new int[8];
        int[] array2 =new int[8];
        long start = System.currentTimeMillis();
        int [] randNum = new int[1000000000];
        for (int i = 0; i < 1000000000; i++) {
            randNum[i] = random.nextInt(Integer.MAX_VALUE);
        }
        System.out.println("随机数生成s耗时"+(System.currentTimeMillis() - start));
        start = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            int temp = randNum[i];
            int temp1 = temp & 7;
            array[temp1] ++;
        }
        System.out.println("&运算耗时"+(System.currentTimeMillis() - start));
        start = System.currentTimeMillis();
        for (int i = 0; i < 1000000000; i++) {
            int temp = randNum[i];;
            int temp2 = temp % 8;
            array2[temp2] ++;
        }
        System.out.println("%运算耗时"+(System.currentTimeMillis() - start));
        System.out.println(Arrays.toString(array));
        System.out.println(Arrays.toString(array2));

运行结果:

随机数生成s耗时16047
&运算耗时842
%运算耗时1135
[125000039, 125000018, 125000011, 125000048, 124999883, 125000270, 124999981, 124999750]
[125000039, 125000018, 125000011, 125000048, 124999883, 125000270, 124999981, 124999750]

我们把数字改成5

随机数生成s耗时15465
&运算耗时1085
%运算耗时1738
[499999525, 0, 0, 0, 500000475]
[200012563, 199986986, 199985673, 199981177, 200033601]

结论:

&位运算速度快于%,缺点:某些奇数如5 时分布不均匀。
联系下另一个方法:tableSizeFor 会将我们传入容量返回为2的倍数。经过实际测试 16 ,32 &位运算比较均匀。

上一篇 下一篇

猜你喜欢

热点阅读