巢狀For迴圈效能優化分析
阿新 • • 發佈:2018-12-22
1、案例描述
某日,在JavaEye上看到一道面試題,題目是這樣的:請對以下的程式碼進行優化
[java] view plain copy print?
- for (int i = 0; i < 1000; i++)
- for (int j = 0; j < 100; j++)
- for (int k = 0; k < 10; k++)
- testFunction (i, j, k);
for (int i = 0; i < 1000; i++) for (int j = 0; j < 100; j++) for (int k = 0; k < 10; k++) testFunction (i, j, k);
2、案例分析
從上述程式碼案例可以看出,不論如何優化,testFunction()執行的次數都是相同的,該部分是不存在優化的可能。那麼優化只能從迴圈變數i,j,k的例項化、初始化、比較、自增等耗時方面來進行分析。首先,分析原題程式碼迴圈變數在以上方面的耗時情況:
變數 | 例項化(次數) | 初始化(次數) | 比較(次數) | 自增(次數) |
i | 1 | 1 | 1000 | 1000 |
j | 1000 | 1000 | 1000*100 | 1000*100 |
k | 1000*100 | 1000*100 | 1000*100*10 | 1000*100*10 |
目的:該程式碼效能優化就是要儘可能地減少迴圈變數i、j、k的例項化、初始化、比較、自增的次數,同時不引起其他可能的耗時運算。
3、解決過程
優化方案①:
[java] view plain copy print?
- for (int i = 0; i < 10; i++)
- for (int j = 0; j < 100; j++)
- for (int k = 0; k < 1000; k++)
- testFunction (k, j, i);
for (int i = 0; i < 10; i++) for (int j = 0; j < 100; j++) for (int k = 0; k < 1000; k++) testFunction (k, j, i);
該方案主要是將迴圈次數少的放在外面,迴圈次數多的放在裡層,這樣可以最大程度地減少相關迴圈變數的例項化次數、初始化次數等,方案耗時情況如下:
變數 | 例項化(次數) | 初始化(次數) | 比較(次數) | 自增(次數) |
i | 1 | 1 | 10 | 10 |
j | 10 | 10 | 10*100 | 10*100 |
k | 10*100 | 10*100 | 10*100*1000 | 10*100*1000 |
優化方案②:
[java] view plain copy print?
- int i, j, k;
- for (i = 0; i < 10; i++)
- for (j = 0; j < 100; j++)
- for (k = 0; k < 1000; k++)
- testFunction (k, j, i);
int i, j, k; for (i = 0; i < 10; i++) for (j = 0; j < 100; j++) for (k = 0; k < 1000; k++) testFunction (k, j, i);
該方案主要是在方案①的基礎上,將迴圈變數的例項化放在迴圈外,這樣可以進一步減少例項化次數,耗時情況如下表:
變數 | 例項化(次數) | 初始化(次數) | 比較(次數) | 自增(次數) |
i | 1 | 1 | 10 | 10 |
j | 1 | 10 | 10*100 | 10*100 |
k | 1 | 10*100 | 10*100*1000 | 10*100*1000 |
注:方案②的優勢體現在若將i、j、k的數值提高更多的,其提升的效果才更明顯。
4、測試程式碼
[java] view plain copy print?
- public class Test {
- public static void main(String[] args){
- // testA();
- // testB();
- testC();
- }
- public static void testA(){
- long start = System.nanoTime();
- for(int i = 0; i < 10; i++)
- for(int j = 0; j < 1000; j++)
- for(int k = 0; k < 10000; k++)
- ;
- System.out.println("testA time>>"+(System.nanoTime()-start)+"ns");
- }
- public static void testB(){
- long start = System.nanoTime();
- for(int i = 0; i < 10000; i++)
- for(int j = 0; j < 1000; j++)
- for(int k = 0; k < 10; k++)
- ;
- System.out.println("testB time>>"+(System.nanoTime()-start)+"ns");
- }
- public static void testC(){
- long start = System.nanoTime();
- int i, j, k;
- for(i = 0; i < 10; i++)
- for(j = 0; j < 1000; j++)
- for(k = 0; k < 10000; k++)
- ;
- System.out.println("testC time>>"+(System.nanoTime()-start)+"ns");
- }
- }