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