C語(yǔ)言從編譯到運(yùn)行過(guò)程詳解
C語(yǔ)言從編譯到運(yùn)行
一、前言
最近在看CSAPP(深入理解計(jì)算機(jī)系統(tǒng))然后以前也學(xué)過(guò)C語(yǔ)言,但是從來(lái)沒(méi)有深究寫(xiě)好的C代碼是怎么編譯再到執(zhí)行的。
所以現(xiàn)在自己學(xué)習(xí),然后記錄下來(lái)。
以最常用的hello world!程序?yàn)槔?程序名: main.c
#include <stdio.h> int main() { printf("Hello world!\n"); return 0; }
二、C程序編譯過(guò)程
hello程序的生命周期是從一個(gè)高級(jí)C語(yǔ)言程序開(kāi)始的,為了能夠運(yùn)行hello.c程序,每一條C語(yǔ)句都被其他程序轉(zhuǎn)化為一系列的低級(jí)機(jī)器語(yǔ)言指令。然后這些指令按照一種稱(chēng)為可執(zhí)行目標(biāo)程序的格式打包,以二進(jìn)制磁盤(pán)文件的形式存放起來(lái)。目標(biāo)程序也稱(chēng)為可執(zhí)行目標(biāo)文件。
編譯一個(gè) C程序可以分為四階段:預(yù)處理階段--->生成匯編代碼階段--->匯編階段--->鏈接階段
各個(gè)階段的代碼可以通過(guò)gcc指令來(lái)生成
如果沒(méi)有g(shù)cc可以用下面指令安裝
sudo apt-get build-dep gcc
安裝完之后可以根據(jù)以下指令查看是否安裝成功
gcc --version
安裝好后用下面指令生成中間文件
gcc main.c 直接生成可執(zhí)行文件 a.out gcc -E main.c -o hello.i 生成預(yù)處理后的代碼 gcc –S main.c -o hello.s 生成匯編代碼 gcc –c main.c -o hello.o 生成目標(biāo)代碼
三、階段過(guò)程
1、預(yù)處理階段
gcc -E main.c -o hello.i 生成預(yù)處理后的代碼
預(yù)處理器(cpp)根據(jù)以字符 # 開(kāi)頭的命令,修改原始的C程序。比如mian.c中第一行的 #include<stdio.h> 命令就告訴預(yù)處理器讀取系統(tǒng)頭文件stdio.h的內(nèi)容,并且把它直接插入程序文本中。同時(shí)刪除注釋行,添加行號(hào)和文件名標(biāo)識(shí)。這樣就得到了另一個(gè)C程序,通常是以 .i 作為文件擴(kuò)展名。 所以經(jīng)過(guò)預(yù)編譯的 .i 文件是不包含宏定義的。
處理完后我們來(lái)看看 hello.i 文件。發(fā)現(xiàn)原來(lái)的7行代碼變成了700多行,我們的代碼在最后面。而前面多出來(lái)的代碼就是 .c 中#include<stdio.h>展開(kāi)的代碼。
2、編譯階段
gcc –S main.c -o hello.s 生成匯編代碼
編譯是將源文件(hello.i)翻譯成匯編文件(hello.s)的過(guò)程。中間包含詞法、語(yǔ)法分析等步驟,具體過(guò)程可以參考《編譯原理》。
打開(kāi)匯編代碼我們會(huì)發(fā)現(xiàn)里面有很多以 . 開(kāi)頭的行,所有這些以 . 開(kāi)頭的行都是指導(dǎo)匯編器和鏈接器工作的偽指令。 我們通??梢院雎赃@些行。
去掉這些行后剩下的部分。
3、匯編階段
gcc –c main.c -o hello.o 生成目標(biāo)代碼
匯編階段是把編譯階段生成的 .s 文件轉(zhuǎn)成 .o 的二進(jìn)制目標(biāo)代碼。匯編器(as)將 hello.s 翻譯成機(jī)器語(yǔ)言指令,把這些指令打包成一種叫做可重定位目標(biāo)程序的格式
,并將結(jié)果保存在目標(biāo)文件hello.o中。hello.o文件是一個(gè)二進(jìn)制文件,它的字節(jié)編碼是機(jī)器語(yǔ)言指令而不是字符。如果我們?cè)谖谋揪幾g器中打開(kāi) hello.o 文件,看到的將是一堆亂碼。
你非要看就是這樣
4、鏈接階段
這個(gè)階段就是把匯編后的機(jī)器指令集變成可以直接運(yùn)行的文件,而對(duì)目標(biāo)文件進(jìn)行鏈接主要是因?yàn)樵谀繕?biāo)文件中可能用到了在其他文件當(dāng)中定義的字段(或者函數(shù)),通過(guò)鏈接來(lái)把多個(gè)不同目標(biāo)文件關(guān)聯(lián)到一起。
hello 程序調(diào)用了printf 函數(shù),它是每個(gè) C 編譯器都會(huì)提供的標(biāo)準(zhǔn)C庫(kù)中的一個(gè)函數(shù),printf 函數(shù)存在于一個(gè)名為 printf.o 的單獨(dú)預(yù)編譯好了的標(biāo)準(zhǔn)文件中,而這個(gè)文件必須以某種方式合并到我們的 hello.o 程序中,鏈接器(ld)就負(fù)責(zé)處理這種合并,結(jié)果就得到 hello 文件,它是一個(gè)可執(zhí)行目標(biāo)文件(簡(jiǎn)稱(chēng):可執(zhí)行文件),可以被加載到內(nèi)存中,有系統(tǒng)執(zhí)行。
以上就是C語(yǔ)言從編譯到運(yùn)行過(guò)程詳解的詳細(xì)內(nèi)容,更多關(guān)于C語(yǔ)言從編譯到運(yùn)行的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Qt串口通信開(kāi)發(fā)之QSerialPort模塊詳細(xì)使用方法與實(shí)例
這篇文章主要介紹了Qt串口通信開(kāi)發(fā)之QSerialPort模塊詳細(xì)使用方法與實(shí)例,需要的朋友可以參考下2020-03-03通過(guò)c語(yǔ)言調(diào)用系統(tǒng)curl動(dòng)態(tài)庫(kù)的示例詳解
這篇文章中我們將通過(guò)一個(gè)簡(jiǎn)單的示例來(lái)講解如何在Ubuntu系統(tǒng)中通過(guò)C語(yǔ)言調(diào)用動(dòng)態(tài)庫(kù)(共享庫(kù))的方法,我們將使用libcurl庫(kù),這是一個(gè)基于客戶(hù)端的URL傳輸庫(kù),廣泛用于各種程序和應(yīng)用中以訪問(wèn)網(wǎng)頁(yè)和服務(wù)器數(shù)據(jù),需要的朋友可以參考下2024-03-03史上最貼心的 VS code C++ 環(huán)境配置超詳細(xì)教程
這篇文章主要介紹了史上最貼心的 VS code C++ 環(huán)境配置超詳細(xì)教程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-02-02C++11中初始化列表initializer lists的使用方法
C++11引入了初始化列表來(lái)初始化變量和對(duì)象,自定義類(lèi)型,如果想用初始化列表就要包含initializer_list頭文件2021-09-09Ubuntu中使用VS Code與安裝C/C++插件的教程詳解
這篇文章主要介紹了Ubuntu中使用VS Code與安裝C/C++插件的教程詳解,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09C語(yǔ)言實(shí)現(xiàn)外賣(mài)管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)外賣(mài)管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11