Math.abs(x)
的正常实现(由Oracle实现)是:
public static double abs(double a) {
return (a <= 0.0D) ? 0.0D - a : a;
}
难道将数字符号的一位编码设置为零(或一)不是更快吗?我认为只有一个比特编码数字的符号,并且它总是相同的比特,但我可能错了。
还是说我们的计算机通常无法使用原子指令对单个位进行操作?
如果可能实现更快的实现,你能给出吗?
编辑:
有人向我指出Java代码是平台无关的,因此它不能依赖于单个机器的原子指令。然而,为了优化代码,JVM热点优化器会考虑机器的具体情况,并且可能应用正在考虑的优化。
然而,通过简单的测试,我发现至少在我的机器上,Math.abs函数似乎没有被优化为单个原子指令。我的代码如下:
long before = System.currentTimeMillis();
int o = 0;
for (double i = 0; i<1000000000; i++)
if ((i-500)*(i-500)>((i-100)*2)*((i-100)*2)) // 4680 ms
o++;
System.out.println(o);
System.out.println("using multiplication: "+(System.currentTimeMillis()-before));
before = System.currentTimeMillis();
o = 0;
for (double i = 0; i<1000000000; i++)
if (Math.abs(i-500)>(Math.abs(i-100)*2)) // 4778 ms
o++;
System.out.println(o);
System.out.println("using Math.abs: "+(System.currentTimeMillis()-before));
这将给我以下输出:
234
using multiplication: 4985
234
using Math.abs: 5587
假设乘法是通过一个原子指令执行的,在我的机器上,JVM hotspot优化器似乎没有将
Math.abs
函数优化为单一指令操作。
abs
比乘法更快。 - Holger