C 的編譯過程

2021-08-31 02:33:45 字數 2760 閱讀 4880

cpp的編譯主要包括四個過程:

1.預編譯

2.**編譯與優化

3.彙編

4.連線

[color=red]經過預編譯[/color]得到的輸出檔案中,只有常量;如數字、字串、變數的定義,以及 c語言的關鍵字,如main,if,else,for,while,, +,-,*,\等等。

[color=red]經過編譯和優化[/color]得到的彙編**必須經過匯程式設計序的彙編轉換成相應的機器指令,方可能被機器執行。

[color=red]彙編過程[/color]實際上指把組合語言**翻譯成目標機器指令的過程

[color=red]鏈結[/color]程式的主要工作就是將有關的目標檔案彼此相連線,也即將在乙個檔案中引用

的符號同該符號在另外乙個檔案中的定義連線起來,使得所有的這些目標檔案成為乙個

能夠誒作業系統裝入執行的統一整體。

********************===

看到const 關鍵字,很多程式設計師想到的可能是const 常量,這可有點象踩到陷井上還不知道自己危險了。讀讀以下文字會使你對c++中的const有乙個全面的認識。

const 是c++中常用的型別修飾符,有某些微妙的應用場合,如果沒有搞清本源,則錯誤在所難免。本篇中將對const進行辨析。溯其本源,究其實質,希望能對大家理解const有所幫助,根據思維的承接關係,分為如下幾個部分進行闡述。

c++中為什麼會引入const

c++的提出者當初是基於什麼樣的目的引入(或者說保留)const關鍵字呢?,這是乙個有趣又有益的話題,對理解const很有幫助。

1. 大家知道,c++有乙個型別嚴格的編譯系統,這使得c++程式的錯誤在編譯階段即可發現許多,從而使得出錯率大為減少,因此,也成為了c++與c相比,有著突出優點的乙個方面。

2. c中很常見的預處理指令 #define variablename variablevalue 可以很方便地進行值替代,這種值替代至少在三個方面優點突出:

一是避免了意義模糊的數字出現,使得程式語義流暢清晰,如下例:

#define user_num_max 107 這樣就避免了直接使用107帶來的困惑。

二是可以很方便地進行引數的調整與修改,如上例,當人數由107變為201時,進改動此處即可,

三是提高了程式的執行效率,由於使用了預編譯器進行值替代,並不需要為這些常量分配儲存空間,所以執行的效率較高。

鑑於以上的優點,這種預定義指令的使用在程式中隨處可見。

預處理語句雖然有以上的許多優點,但它有個比較致命的缺點,即,預處理語句僅僅只是簡單值替代,缺乏型別的檢測機制。這樣預處理語句就不能享受 c++嚴格型別檢查的好處,從而可能成為引發一系列錯誤的隱患。

4.好了,第一階段結論出來了:

結論: const 推出的初始目的,正是為了取代預編譯指令,消除它的缺點,同時繼承它的優點。

現在它的形式變成了:

const datatype variablename = variablevalue ;

為什麼const能很好地取代預定義語句?

const 到底有什麼大神通,使它可以振臂一揮取代預定義語句呢?

1. 首先,以const 修飾的常量值,具有不可變性,這是它能取代預定義語句的基礎。

2. 第二,很明顯,它也同樣可以避免意義模糊的數字出現,同樣可以很方便地進行引數的調整和修改。

3. 第三,c++的編譯器通常不為普通const常量分配儲存空間,而是將它們儲存在符號表中,這使得它成為乙個編譯期間的常量,沒有了儲存與讀記憶體的操作,使得它的效率也很高,同時,這也是它取代預定義語句的重要基礎。

這裡,我要提一下,為什麼說這一點是也是它能取代預定義語句的基礎,這是因為,編譯器不會去讀儲存的內容,如果編譯器為const分配了儲存空間,它就不能夠成為乙個編譯期間的常量了。

4. 最後,const定義也像乙個普通的變數定義一樣,它會由編譯器對它進行型別的檢測,消除了預定義語句的隱患。

const 使用情況分類詳析

1.const 用於指標的兩種情況分析:

int const *a; //a/可變,*a不可變

int *const a; //a不可變,*a可變

分析:const 是乙個左結合的型別修飾符,它與其左側的型別修飾符合為乙個型別

修飾符,所以,int const 限定 *a,不限定a。int *const 限定a,不限定*a。

2.const 限定函式的傳遞值引數:

void fun(const int var);

分析:上述寫法限定引數在函式體中不可被改變。由值傳遞的特點可知,var在函式體中的改變不會影響到函式外部。所以,此限定與函式的使用者無關,僅與函式的編寫者有關。

結論:最好在函式的內部進行限定,對外部呼叫者遮蔽,以免引起困惑。如可改寫如下:

void fun(int var)

3.const 限定函式的值型返回值:

const int fun1();

const myclass fun2();

分析:上述寫法限定函式的返回值不可被更新,當函式返回內部的型別時(如fun1),已經是乙個數值,當然不可被賦值更新,所以,此時const無意義,最好去掉,以免困惑。當函式返回自定義的型別時(如fun2),這個型別仍然包含可以被賦值的變數成員,所以,此時有意義。

5. const 限定類的成員函式:

class classname

注意:採用此種const 後置的形式是一種規定,亦為了不引起混淆。在此函式的宣告中和定義中均要使用const,因為const已經成為型別資訊的一部分。

獲得能力:可以操作常量物件。

失去能力:不能修改類的資料成員,不能在函式中呼叫其他不是const的函式。

c編譯過程

編譯的概念 編譯程式讀取源程式 字元流 對之進行詞法和語法的分析,將高階語言指令轉換為功能等效的彙編 再由匯程式設計序轉換為機器語言,並且按照作業系統對可執行檔案格式的要求鏈結生成可執行程式。編譯的完整過程 c源程式 預編譯處理 c 編譯 優化程式 s asm 匯程式設計序 obj o a ko 鏈...

c 編譯過程

編譯過程主要分為 4個過程 1 編譯預處理 預編譯程式完成的工作,可以說成是對源程式的 替換 工作。經過這個過程,生成乙個沒有巨集定義 沒有條件編譯指令 沒有特殊符號的輸出檔案。2 編譯 優化階段 通過詞法分析 語法分析,在確認所有的指令都符合語法規則之後,將其翻譯成等價的中間 或彙編 在c 中,以...

C 編譯過程

以helloword.c 程式說明編譯過程 在預設的狀態下,如果我們直接以gcc編譯原始碼,並且沒有加上任何引數,則執行檔案的檔名會被自動設定為a.out 這個檔名。所以你就能夠直接執行 a.out這個這行檔案。hello.c 就是原始碼,gcc是編譯程式,a.out 是編譯成功的可執行檔案。如果我...