当我检查这个问题时,我不相信它,所以我测试了,似乎是真的.内部循环中的声明似乎比声明外部循环更快.有人可以解释为什么会这样吗?
这是我的测试代码:
public class CycleTest {
/**
* @param args the command line arguments
*/
public static void main(String[] args) {
long iterations = 1000000;
warmUp(iterations);
System.out.println("Cycle1");
double individualTime = getAverageTimePerIterationc1(iterations);
iterations = 1000;
double totalTime = getTotalTimec1(iterations);
System.out.println("ns/iteration: " + individualTime);
System.out.println("Total time for " + iterations + " runs: " + totalTime);
System.out.println("Cycle2");
iterations = 1000000;
double individualTime1 = getAverageTimePerIterationc2(iterations);
iterations = 1000;
double totalTime1 = getTotalTimec2(iterations);
System.out.println("ns/iteration: " + individualTime1);
System.out.println("Total time for " + iterations + " runs: " + totalTime1);
}
public static void warmUp(long iterations) {
System.out.println("Starting warmup");
for (int i = 0; i < iterations; i++) {
runCycles();
runCycles1();
}
}
public static double getAverageTimePerIterationc1(long iterations) {
// test
System.out.println("Starting individual time test");
long timeTaken = 0;
for (int i = 0; i < iterations; i++) {
long startTime = System.nanoTime();
runCycles();
timeTaken += System.nanoTime() - startTime;
}
return (double) timeTaken / iterations;
}
public static long getTotalTimec1(long iterations) {
// test
System.out.println("Starting total time test");
long timeTaken = 0;
for (int i = 0; i < iterations; i++) {
long startTime = System.nanoTime();
runCycles();
timeTaken += System.nanoTime() - startTime;
}
return timeTaken;
}
public static double getAverageTimePerIterationc2(long iterations) {
// test
System.out.println("Starting individual time test");
long timeTaken = 0;
for (int i = 0; i < iterations; i++) {
long startTime = System.nanoTime();
runCycles1();
timeTaken += System.nanoTime() - startTime;
}
return (double) timeTaken / iterations;
}
public static long getTotalTimec2(long iterations) {
// test
System.out.println("Starting total time test");
long timeTaken = 0;
for (int i = 0; i < iterations; i++) {
long startTime = System.nanoTime();
runCycles1();
timeTaken += System.nanoTime() - startTime;
}
return timeTaken;
}
private static void runCycles() {
double intermediateResult;
for (int i = 0; i < 1000; i++) {
intermediateResult = i;
intermediateResult += 1;
}
}
private static void runCycles1() {
for (int i = 0; i < 1000; i++) {
double intermediateResult = i;
intermediateResult += 1;
}
}
}
Run Code Online (Sandbox Code Playgroud)
额外的新信息:我在Windows机器上的JDK 1.6.0_27上运行它.
它们生成相同的代码.这两种方法:
private static void runCycles1() {
double intermediateResult;
for (int i = 0; i < 1000; i++) {
intermediateResult = i;
}
}
private static void runCycles2() {
for (int i = 0; i < 1000; i++) {
double intermediateResult = i;
}
}
Run Code Online (Sandbox Code Playgroud)
在Java 8(jdk1.8.0_51)中生成此字节码:
runCycles1() runCycles2()
Code: Code:
0: iconst_0 0: iconst_0 i = 0
1: istore_2 1: istore_0
2: goto 11 2: goto 11 goto 11
5: iload_2 5: iload_0 intermediateResult = (double)i
6: i2d 6: i2d
7: dstore_0 7: dstore_1
8: iinc 2, 1 8: iinc 0, 1 i++
11: iload_2 11: iload_0 if (i < 1000) goto 5
12: sipush 1000 12: sipush 1000
15: if_icmplt 5 15: if_icmplt 5
18: return 18: return return
Run Code Online (Sandbox Code Playgroud)
实际的声明不生成任何代码,所以如果你看到性能上的差异,我会感到惊讶.
所以测试了它:
for (int j = 0; j < 10; j++) {
long t1 = System.nanoTime();
for (int i = 0; i < 1000_000_000; i++)
runCycles1();
long t2 = System.nanoTime();
for (int i = 0; i < 1000_000_000; i++)
runCycles2();
long t3 = System.nanoTime();
System.out.printf("%d %d%n", t2 - t1, t3 - t2);
}
Run Code Online (Sandbox Code Playgroud)
得到了:
4250095 2020120
4067898 0
3904236 0
0 0
0 0
0 0
0 0
0 0
0 0
0 0
Run Code Online (Sandbox Code Playgroud)
他们最终都编译成没有!