wtf???Java乘法会比位运算快??你一定是在逗我!!哈哈哈,当时也困扰了我,让我们继续向下看。

我们学习c的时候,都会遇到位运算快于乘法运算的情况,当时老师讲的是计算机只有加法运算,乘法运算的运算需要多次的进位和转换,所以位运算的效率要远远高于乘法运算。

管中窥豹,将相同的情况拿到了java中测试了一下,结果却和预想的不太一样,在相同的循环次数下,乘法运算和位运算的时间效率是交替着的,也就是说位运算和乘法运算有相近的计算效率。我们看下文来具体分析。

一、样例

Talk is weak, show me the code!
我们首先来看代码,各位看官可以直接拿下来跑一下。

public static void main(String []args){
        int a=2;
        int b=2;
        int aResult;
        int bResult;
        long startTime;
        long endTime;
        long time;
		
		//乘法运算
        startTime=System.currentTimeMillis();
        for(int i=0;i<250000000;i++){
            bResult=b*1024;
        }
        endTime=System.currentTimeMillis();
        time=endTime-startTime;
        System.out.println("乘法250000000次时间:"+time);
       
       //位运算
		System.out.println(Integer.toBinaryString(1024));
        startTime=System.currentTimeMillis();
        for(int i=0;i<250000000;i++){
            aResult=a<<10;
        }
        endTime=System.currentTimeMillis();
        time=endTime-startTime;
        System.out.println("位移100000000次时间:"+time);
    }

二、 结果

  • 运算结果一:
  • java int乘法_System

  • 运算结果二:
  • java int乘法_java int乘法_02

三、位运算的过程

上面的例子中,我们有第一句:

aResult=a<<10

这就是我们所说的位运算。
等价于:

bResult=b*1024;

其中我们输出了1024的二进制形式:

System.out.println(Integer.toBinaryString(1024));

结果为100000000000
最高位为第10位(从0开始计数),因此向左移位10位等同于数值乘1024。
所以两句话 (乘1024于<<10)是等价的
因此具有可比性。

四、产生这种情况的原因?

最开始我对出现的这种现象也产生了很大的疑惑,尤其是对于Java的乘法竟然有时候会比位运算还要快!
????wtf!
后来我对产生这种情况的原因进行了搜索,发现很少有对底层的计算过程进行讲解的文章,大部分是人云亦云,亦有很多是直接拷贝的别人的博客,而且不加验证!!我想说,你们这样误导别人真的好吗!!!真的很痛心,虽然个人感觉对别人的知识进行摘取并不可耻,但也要加以验证,并从中产生自己的思考,这才是真正有意义的,并且对别人也是极有帮助的。每个人人云亦云,真的是 **!

初步猜测:

  1. java在计算乘法的时候一定采取了优化
    不然按照之前的理论和实际,位运算肯定是要比乘法要慢得多,所以一定是java在计算时对过程进行了改造。
  2. 那么采用了什么优化呢?
    个人猜测java在底层将乘法转化成了位运算,所以会有和位运算相近的运算时间。也就是上面我们说的运行时间总是在1234之间浮动,且大部分为2(运行时间根据不同的机器会有存在一定的差异,不能一概而论)。
  3. 那么为什么会产生有时快有时慢的交叉的情况呢?
    这就要从操作系统的底层思考了。我们都知道类似windows,linux,macos都是分时操作系统,cpu计算快,而数据的传输较cpu的运算要慢上万的数量级,因此在操作系统为每一个进行分配了时间片,在运行时间片为0时,便把该进程切换下来,换下一个进程,这也就是为什么我们能够同时运行许多不同的程序的原因。(快到我们以为程序是在同时运行着)
    在我们运行测试样例的时候,产生了cpu的调度和进程切换,电脑的cpu分配了时间给其他的进程,影响了速度,所以会产生数值上少量的差异。