自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

性能優(yōu)化小技巧-消除低效循環(huán),讓你的程序快到飛起

開發(fā) 前端
由于編譯器的優(yōu)化非常小心,它必須確保優(yōu)化前后執(zhí)行的效果是保持一致的,因此有些時候它會變得保守,并不能幫你優(yōu)化太多。本文所需要的是在平常不需要花費太多力氣,養(yǎng)成習(xí)慣,并且對程序性能有好處的小技巧。

在分享這些性能優(yōu)化技巧之前,需要說明以下幾點:

  • 不要過早優(yōu)化性能
  • 現(xiàn)代編譯器的優(yōu)化能力很強大
  • 80%的性能問題集中于20%的代碼中

但是由于編譯器的優(yōu)化非常小心,它必須確保優(yōu)化前后執(zhí)行的效果是保持一致的,因此有些時候它會變得保守,并不能幫你優(yōu)化太多。

本文所需要的是在平常不需要花費太多力氣,養(yǎng)成習(xí)慣,并且對程序性能有好處的小技巧。

[[322875]]

示例程序

為了說明本文所提到的技巧效果,先看一個示例程序,程序的目的非常簡單,就是將字符串中的小寫字母轉(zhuǎn)換為大寫),以下是完整可編譯運行代碼:

  1. #include<stdlib.h> 
  2. #include<stdio.h> 
  3. #include<time.h> 
  4. #include<ctype.h> 
  5. #include<string.h> 
  6. #include<sys/time.h> 
  7. #define MAX_LEN  1024*1024 
  8. void printCostTime(struct timeval *start,struct timeval *end) 
  9.     if(NULL == start || NULL == end) 
  10.     { 
  11.         return; 
  12.     } 
  13.     long cost = (end->tv_sec - start->tv_sec) * 1000 + (end->tv_usec - start->tv_usec)/1000; 
  14.     printf("cost time: %ld ms\n",cost); 
  15. int main(void) 
  16.     srand(time(NULL)); 
  17.  
  18.     int min = 'a'
  19.     int max = 'z'
  20.     char *str = malloc(MAX_LEN); 
  21.     //申請失敗則退出 
  22.     if(NULL == str) 
  23.     { 
  24.         printf("failed\n"); 
  25.         return -1; 
  26.     } 
  27.     unsigned int i = 0
  28.     while(i < MAX_LEN)//生成隨機(jī)數(shù) 
  29.     { 
  30.         str[i] = ( rand() % ( max - min ) ) + min; 
  31.         i++; 
  32.     } 
  33.     str[MAX_LEN - 1] = 0;  
  34.     //統(tǒng)計時間 
  35.     struct timeval start,end; 
  36.     gettimeofday(&start,NULL); 
  37.     for(i = 0;i < strlen(str) ;i++) 
  38.     { 
  39.         str[i]  = toupper( str[i] ); 
  40.     } 
  41.     gettimeofday(&end,NULL); 
  42.     printCostTime(&start,&end); 
  43.     free(str); 
  44.     str = NULL
  45.     return 0; 

隨機(jī)數(shù)的生成可參考《隨機(jī)數(shù)生成的方法》。我們主要關(guān)注下面的部分:

  1. for(i = 0;i < strlen(str) ;i++) 
  2.     str[i]  = toupper( str[i] ); 

很簡單,對不對?

運行看看時間:

  1. $ gcc - -o loop loop.c 
  2. $ ./loop 
  3. cost time: 42103 ms 

總共花了42秒多!(機(jī)器處理能力不同運行結(jié)果將會有較大差異)

消除低效循環(huán)

終于來到了我們的優(yōu)化環(huán)節(jié),我們觀察代碼其實很容易發(fā)現(xiàn),每次循環(huán)的時候都會執(zhí)行一次strlen計算字符串的長度,而這個計算具有以下特點

每次結(jié)果一致,屬于重復(fù)計算

strlen時間復(fù)雜度為O(N),也就是說,字符串越長,它需要的時間也就越多

一般情況下的使用是沒有太大問題的,但是問題在于,如果是在一個多次循環(huán)中,它能極大的影響效率。

到這里,優(yōu)化方法想必你也清楚了,那就是將計算結(jié)果不會改變的計算移到循環(huán)外。代碼如下:

  1. unsigned int len = strlen(str); 
  2. for(i = 0;i < len ;i++) 
  3.     str[i]  = toupper( str[i] ); 

那么再次運行的結(jié)果如何呢?

  1. $ gcc -O0 -o loop loop.c 
  2. $ ./loop 
  3. cost time: 4 ms 

看到?jīng)]有,4ms,將近一萬的性能提升!而這個數(shù)值將會隨著字符串長度的增長進(jìn)一步擴(kuò)大。驚不驚喜,意不意外?

總結(jié)

實際上,本文的例子是比較極端的,然后實際中就可能隱藏著很多類似的代碼:

  • 在循環(huán)中計算,但是每次結(jié)果都一樣
  • 并且該計算的復(fù)雜度不是O(1)

對于這類代碼,在不絕對影響可讀性的情況下,完全可以將其移到循環(huán)外。

思考

如果是C++的string,循環(huán)時通過str.length()獲取長度,會如此影響性能嗎?為什么?

 

責(zé)任編輯:趙寧寧 來源: 編程珠璣
相關(guān)推薦

2024-06-12 12:28:23

2020-11-18 13:54:27

IDEAJava開發(fā)

2021-01-04 15:11:57

開發(fā) IDEA代碼

2020-05-06 16:32:18

for循環(huán)Python迭代

2024-11-27 09:46:34

2023-03-01 23:59:23

Java開發(fā)

2024-06-21 08:21:44

2025-04-15 00:00:00

2015-09-16 14:47:14

Android性能優(yōu)化代碼

2023-09-26 12:02:34

C++循環(huán)

2019-07-18 12:40:49

Java編程語言性能優(yōu)化

2021-11-18 08:20:22

接口索引SQL

2020-09-29 07:54:05

Express 飛起

2011-04-13 10:51:58

MATLAB

2017-03-15 09:42:43

軟件開發(fā)云攜手

2025-04-09 00:01:05

2022-02-21 13:27:11

接口性能優(yōu)化索引命令

2024-01-08 17:09:07

Python解釋器CPython

2011-05-20 11:12:01

數(shù)據(jù)庫DB2優(yōu)化
點贊
收藏

51CTO技術(shù)棧公眾號