成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

優化C++代碼(3)常量合并

開發 后端
這篇文章講的是常量合并,這是VC++編譯器最簡單的優化之一。 這種優化,是指編譯器在編譯時(編譯期間)直接計算出表達式的結果,在生成的代碼中直接用計算結果替換表達式。 這樣就避免了程序在運行時執行這些計算花費的成本。

這篇文章講的是常量合并,這是VC++編譯器最簡單的優化之一。  這種優化,是指編譯器在編譯時(編譯期間)直接計算出表達式的結果,在生成的代碼中直接用計算結果替換表達式。 這樣就避免了程序在運行時執行這些計算花費的成本。

下面是一個例子 APP.cpp文件中的 main 函數:

  1. int main() { return 7 + 8; } 

首先,關于這篇文章的一些須知:

  1. 我們將從命令行來構建程序(而不是Visual Studio)
  2. 我們會使用Visual Studio 2012。 特別注意的是,這個版本的編譯器會產生x64位代碼(而不是已經過時的x86架構)在64位機子上編譯。

如果你想要繼續,請看下說明。實際上,你只需要從Visual Studio 列表里選擇一個正確的變體。

(注意:如果你正在使用Visual Studio Express上的免費編譯器,它僅僅只能運行在x86上,但是也會順利生成x64的代碼。對這個實驗同樣有用。)

我們可以通過命令 CL /FA App.cpp來構建示例程序。用/FA開關創建一個輸出文件,用來保存編譯器生成的匯編代碼,可以輸入type App.asm來顯示:

  1. PUBLIC  main 
  2. _TEXT   SEGMENT 
  3. main    PROC 
  4.         mov     eax, 15 
  5.         ret     0 
  6. main    ENDP 
  7. _TEXT   ENDS 
  8. END 

有趣的是這條指令 move ax,15—-僅僅將15賦值給寄存器EAX(根據x64調用標準的定義,x64函數將會設置一個int值,作為函數的結果,并返回給調用者)。編譯器運行期間并沒有發出 7加8的指令。就像下面這樣:

  1. PUBLIC  main 
  2. _TEXT   SEGMENT 
  3. main    PROC 
  4.         mov     eax, 7 
  5.         add     eax, 8 
  6.         ret     0 
  7. main    ENDP 
  8. _TEXT   ENDS 
  9. END 

(注意看了,這兩段代碼的***一條指令,ret 0,是指將控制權返回給調用者,并從棧里彈出0個字節。不要被誤導認為是返回數值0給調用者!)

我猜到,你可能在想:這很好啊,但是哪個白癡會想到在代碼里寫 7+8 這樣的運算?的確,你是對的,但是編譯器會把這樣的結構看成是有副作用的宏。看了下面的例子,你就會明白常量合并是一個很有用的優化方法:

  1. #define SECS_PER_MINUTE  60 
  2. #define MINUTES_PER_HOUR 60 
  3. #define HOURS_PER_DAY    24 
  4.   
  5. enum Event { Started, Stopped, LostData, ParityError }; 
  6.   
  7. struct { 
  8.     int        clock_time; 
  9.     enum Event ev; 
  10.     char*      reason; 
  11. }   Record; 
  12.   
  13. int main() { 
  14.     const int table_size = SECS_PER_MINUTE * MINUTES_PER_HOUR * HOURS_PER_DAY * sizeof Record; 
  15.     // rest of program 

我們要創建一個足夠大的表保存每一秒的記錄,所以table_size就是表的大小,用字節表示。很容易查看變量table_size的匯編指令:

  1. mov     DWORD PTR table_size$[rsp], 1382400     ; 00151800H 

這兒沒有乘法指令,60*60*24*16=1382400 是在編譯時計算的。

事實上,我們窺探下編譯器的內部,會發現這種常量合并的運算非常簡單,它是由前端來執行的。它并不需要后端優化器笨重的提升能力。所以它總是存在的。不管你是開啟優化(使用 /O2)或者關閉優化(/Od)都沒什么區別—–該優化總是自動執行的。

不管表達式有多復雜,我們都能在編譯期間進行常量合并嗎?—事實上,前端可以處理任意的常量算術表達式(甚至包括上面提到的sizeof,只要它們在編譯時能被計算出來)和運算符(+ - * / % << >> ++ 和 –)。你甚至可以使用布爾值,邏輯運算符 和條件運算符if AND ?:。

有沒有常量合并需要后端優化器的時候呢?當然有,看下面的例子:

  1. int bump(int n) { return n + 1; }  
  2.    
  3. int main() { return 3 + bump(6); }  

輸入命令cl /FA /Od App.cpp,會得到信息:不能優化,謝謝!,輸入 App.asm,我們會得到:

  1. mov     ecx, 6 
  2. call    ?bump@@YAHH@Z                           ; bump 
  3. add     eax, 3 

正如我們所預料的: ECX會保存***個參數6,根據x64調用約定,然后調用bump函數,結果返回給EAX,然后EAX再加3。

我們來看看如果我們使用 cl /FA /O2 App.cpp 來進行優化,會發生什么。

  1. mov eax,10 

后端優化器已經識別到bump函數很小,可以包含到調用者里(我們在后面的章節將會講到這種優化方法,叫做內聯函數)。它在編譯時就能夠估算出整個表達式的值,***只剩下一條單指令。很神奇,對吧?

原文鏈接:http://blogs.msdn.com/b/vcblog/archive/2013/07/04/optimizing-c-code-constant-folding.aspx

譯文鏈接:http://blog.jobbole.com/47191/

責任編輯:陳四芳 來源: 伯樂在線
相關推薦

2013-09-05 09:50:11

C++代碼優化

2010-01-13 13:27:00

C++優化

2024-01-25 16:19:27

2013-09-03 09:35:10

2010-02-05 17:49:24

C++常量引用

2024-01-25 11:42:00

C++編程指針常量

2020-07-19 10:23:13

C++進制常量

2023-11-15 17:58:58

C++代碼

2023-10-30 10:29:50

C++最小二乘法

2021-06-10 09:40:12

C++性能優化Linux

2011-07-12 11:15:46

C++

2010-01-14 14:40:21

C++代碼

2010-01-18 16:17:53

C++代碼

2013-09-03 09:30:42

C++代碼優化

2011-05-18 17:56:38

C#C++

2011-05-18 18:05:47

C#C++

2010-01-14 16:35:31

C++優化

2010-01-21 10:23:53

C++代碼

2010-01-18 13:42:51

C++代碼

2010-02-02 15:59:32

C++賦值函數
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 九七午夜剧场福利写真 | 亚洲精品短视频 | a免费在线 | 蜜臀久久99精品久久久久野外 | 亚洲高清视频在线观看 | 欧美一级在线 | 国产视频一区二区在线观看 | 日韩精品一区二区三区视频播放 | 久久不卡日韩美女 | 成人av免费 | 拍真实国产伦偷精品 | 国产精品日韩一区 | 一区二区精品在线 | 一区二区三区视频在线观看 | 亚洲国产精品久久 | 91精品久久久久久久久中文字幕 | 成人一区二| 久久偷人| 中文字幕精品一区久久久久 | 亚洲在线一区二区 | 国产成人精品久久二区二区91 | 美女爽到呻吟久久久久 | 免费黄色av网站 | 99久久99久久精品国产片果冰 | 中文日韩字幕 | 久久久蜜桃一区二区人 | 成人二区| 午夜影院视频 | 成人精品啪啪欧美成 | 欧美 日韩 亚洲91麻豆精品 | 国产午夜精品一区二区三区嫩草 | 精品乱码一区二区 | 91视视频在线观看入口直接观看 | 国产精品色 | 亚洲精品久久久 | 国产成人一区二区三区久久久 | 久久国产精品偷 | 欧日韩在线 | 国产麻豆乱码精品一区二区三区 | 成人av网站在线观看 | 羞羞视频网站免费看 |