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

Gcc/G++/Gdb 的正確打開方式:從編譯到調(diào)試,一次搞懂!

開發(fā) 前端
gcc,全稱 GNU Compiler Collection,是一款強(qiáng)大的開源編譯器,支持多種語言(C、C++、Objective-C 等)。但今天,我們只專注它在 C/C++ 編譯領(lǐng)域的表現(xiàn)。

大家好,我是小康。今天我們來聊下怎樣來編譯和調(diào)試 C/C++ 程序。

提到 gcc/g++,很多初學(xué)者的第一反應(yīng)可能是:“這不是編譯器嘛,不就寫 gcc main.c 然后敲個(gè)回車?”但當(dāng)編譯報(bào)錯(cuò)時(shí),才發(fā)現(xiàn)自己對(duì)它的了解就像對(duì)前任一樣——一知半解。

其實(shí),gcc/g++ 不只是一個(gè)“會(huì)把 C/C++ 代碼變成可執(zhí)行文件”的工具,它還能優(yōu)化、調(diào)試、排錯(cuò),甚至分析代碼!今天,小康就來帶你解鎖 gcc/g++/gdb 的正確姿勢(shì):從編譯到調(diào)試,通通搞明白!

小貼士:

  • gcc:C 編譯器,專門編譯 C 程序。
  • g++ : C++編譯器,專門編譯 C++ 程序。gcc 和 g++的用法和參數(shù)基本相同,今天我們主要介紹 gcc!
  • gdb :調(diào)試 C/C++ 程序的利器!

一、什么是 gcc?簡(jiǎn)單聊聊它的身份

gcc,全稱 GNU Compiler Collection,是一款強(qiáng)大的開源編譯器,支持多種語言(C、C++、Objective-C 等)。但今天,我們只專注它在 C/C++ 編譯領(lǐng)域的表現(xiàn)。

一句話概括 gcc 的工作:把你寫的代碼從“人話”翻譯成機(jī)器能看懂的“機(jī)器語言”。即:將你的程序代碼編譯成計(jì)算機(jī)能夠識(shí)別的機(jī)器語言(01機(jī)器碼)。

gcc 的核心流程分為四步:

  1. 預(yù)處理:處理宏定義、頭文件、條件編譯等。
  2. 編譯:將預(yù)處理的代碼轉(zhuǎn)成匯編代碼。
  3. 匯編:把匯編代碼轉(zhuǎn)成機(jī)器代碼(生成目標(biāo)文件)。
  4. 鏈接:將目標(biāo)文件生成可執(zhí)行文件。

二、GCC 的安裝與檢查

2.1 檢查是否已安裝

在終端輸入以下命令:

gcc --version

如果返回版本信息,說明 GCC 已經(jīng)安裝成功。如果提示 command not found,那就繼續(xù)看下面的安裝步驟。

2.2 安裝 GCC/G++/GDB

  • Ubuntu/Debian 系統(tǒng):
sudo apt update
sudo apt install build-essential -y   # 安裝 gcc 和 g++
sudo apt install gdb                  # 安裝 gdb

這會(huì)同時(shí)安裝 GCC 和其他編譯工具鏈。

  • CentOS/Red Hat 系統(tǒng):
sudo yum groupinstall "Development Tools" -y  # 安裝 gcc 和 g++

sudo yum install gdb -y                       # 安裝 gdb
  1. 驗(yàn)證安裝:分別運(yùn)行 gcc --version 、 g++ --version和 gdb --version,確認(rèn) GCC/G++/GDB 是否安裝成功。

三、gcc 的基本用法:從入門到熟練

3.1 最簡(jiǎn)單的編譯指令

gcc main.c -o main
  • main.c 是你的代碼文件。
  • -o main 指定生成的可執(zhí)行文件名為 main。如果不寫 -o,默認(rèn)生成名為 a.out 的文件。

運(yùn)行程序:

./main

就這么簡(jiǎn)單,一行命令搞定編譯和運(yùn)行,是不是挺方便?但這其實(shí)是“打包式”的操作,編譯和鏈接一起完成。如果你是剛?cè)腴T的初學(xué)者,可能還不知道 GCC 背后做了些什么。這時(shí),我們可以試試 分步編譯,讓每一步變得更清晰。

3.2 分步編譯指令

分步編譯可以幫你更好地理解編譯器的工作流程。其實(shí),GCC 編譯分為兩個(gè)主要階段(G++ 類似):

  1. 編譯階段:將源碼翻譯成機(jī)器能理解的中間文件(目標(biāo)文件,.o 文件)。
  2. 鏈接階段:將目標(biāo)文件鏈接成最終的可執(zhí)行文件。

第一步:編譯源程序文件

運(yùn)行以下命令,將 main.c 轉(zhuǎn)換成目標(biāo)文件 main.o:

gcc -c main.c -o main.o
  • -c 參數(shù)表示只編譯,不鏈接。
  • main.o 是生成的目標(biāo)文件,雖然不能直接運(yùn)行,但它已經(jīng)包含了 main.c 的翻譯結(jié)果。

第二步:鏈接目標(biāo)文件

接下來,將目標(biāo)文件 main.o 鏈接成可執(zhí)行文件 main:

gcc main.o -o main
  • 這一步不再使用 -c,而是使用 -o,因?yàn)槲覀円?GCC 把目標(biāo)文件鏈接成一個(gè)完整的程序。

運(yùn)行程序:

./main

同樣的輸出,經(jīng)過分步操作生成了結(jié)果,是不是感覺自己更專業(yè)了?

為什么要分步編譯?

你可能會(huì)想:“分兩步多麻煩啊,我直接一步編譯不就行了?”其實(shí),分步編譯有它的優(yōu)勢(shì):

  • 更高的靈活性:當(dāng)項(xiàng)目中有多個(gè)文件時(shí),你只需要重新編譯修改過的文件,其他部分可以直接復(fù)用之前生成的目標(biāo)文件(.o文件 ),大大提高效率。(下文會(huì)提到多個(gè)文件編譯的情況)
  • 清晰的流程:每一步的工作職責(zé)都很明確,便于排查問題。例如,如果某個(gè)文件編譯不過,可以單獨(dú)解決,而不用從頭來過。

3.2 常用選項(xiàng)大盤點(diǎn)

  • 加點(diǎn)料,讓錯(cuò)誤信息更清晰:
gcc -Wall -Wextra main.c -o main

-Wall:開啟常見警告(比如變量聲明但沒使用 -Wunused-variable  )

-Wextra:開啟額外警告(如未使用函數(shù)參數(shù) -Wunused-parameter)

  • 為調(diào)試準(zhǔn)備,加上調(diào)試符號(hào):
gcc -g main.c -o main

-g:生成調(diào)試信息,方便用 GDB 調(diào)試。

  • 編譯多個(gè)文件:
gcc file1.c file2.c -o program

多個(gè)源文件會(huì)一起編譯鏈接成一個(gè)可執(zhí)行文件。

  • 優(yōu)化代碼(-O 系列)

讓程序更快?試試優(yōu)化選項(xiàng):

-O0:不優(yōu)化(默認(rèn))

gcc 默認(rèn)不會(huì)優(yōu)化代碼,生成的程序跟你寫的源代碼更接近。

啥時(shí)候用?

  • 開發(fā)調(diào)試時(shí),容易追蹤代碼邏輯。

編譯示例:

# 這兩個(gè)命令效果一樣
gcc hello.c -O0 -o hello
gcc hello.c -o hello

-O1:基礎(chǔ)優(yōu)化

會(huì)優(yōu)化掉無用代碼,讓程序稍微跑快一點(diǎn),但調(diào)試依然友好。

啥時(shí)候用?

  • 需要一點(diǎn)性能提升,但還得經(jīng)常調(diào)試代碼的時(shí)候。

編譯示例:

gcc hello.c -O1 -o hello

-O2:常用優(yōu)化(推薦?。?/p>

在 -O1 的基礎(chǔ)上,增加更多優(yōu)化,比如減少循環(huán)次數(shù)、改進(jìn)分支預(yù)測(cè)等。

啥時(shí)候用?

  • 程序跑得還可以,但希望它跑得更穩(wěn)更快。適合大部分場(chǎng)景。

編譯示例:

gcc hello.c -O2 -o hello

-O3:更高級(jí)優(yōu)化

比 -O2 更激進(jìn),開啟一些高級(jí)優(yōu)化,比如函數(shù)內(nèi)聯(lián)和向量化。

啥時(shí)候用?

  • 追求極限性能的程序,比如科學(xué)計(jì)算、大型數(shù)據(jù)處理。但注意,有時(shí)候優(yōu)化過頭會(huì)導(dǎo)致兼容性問題(比如浮點(diǎn)運(yùn)算不準(zhǔn))。

編譯示例:

gcc hello.c -O3 -o hello

總結(jié):選擇適合的優(yōu)化級(jí)別

  • 開發(fā)階段:-O0 或 -O1,方便調(diào)試。
  • 生產(chǎn)環(huán)境:-O2 是最平衡的選項(xiàng),跑得快又穩(wěn)。
  • 極限性能:-O3 ,但要注意兼容性和精度問題。

根據(jù)場(chǎng)景選個(gè)合適的優(yōu)化級(jí)別,你的代碼就能跑得既穩(wěn)又快!

四、多文件項(xiàng)目的編譯

在實(shí)際項(xiàng)目中,代碼往往分成多個(gè)文件,比如:

  • main.c
  • utils.c
  • utils.h

方法一:一次性編譯

gcc main.c utils.c -o my_program

優(yōu)勢(shì):

  • 簡(jiǎn)單粗暴:一條命令搞定所有文件,適合小項(xiàng)目。
  • 快速省事:文件少的時(shí)候,用起來方便快捷。

方法二:分步編譯再鏈接

gcc -c main.c -o main.o  
gcc -c utils.c -o utils.o  
gcc main.o utils.o -o my_program

優(yōu)勢(shì):

  • 效率高:只編譯修改過的文件,不用每次全編譯。
  • 更靈活:大項(xiàng)目用分步編譯更好管理,還能配合自動(dòng)化工具用。

建議:

文件少就用方法一,文件多或者想提高效率就用方法二,兩個(gè)都得會(huì),隨場(chǎng)景切換!

五、gcc 編譯流程深入解析:搞懂每一步

如果只知道用 gcc 編譯,算是入門;但要真正搞懂 gcc,必須了解它的四步工作流程。

5.1 預(yù)處理:先搞定頭文件和宏

gcc -E main.c -o main.i
  • -E:只執(zhí)行預(yù)處理,輸出結(jié)果是 main.i。
  • 預(yù)處理會(huì)替換 #include 的頭文件內(nèi)容,展開宏定義,去掉注釋。
  • 打開生成的文件,你能看到“裸露”的預(yù)處理后代碼。

5.2 編譯:從人話到匯編

gcc -S main.i -o main.s
  • -S:將預(yù)處理后的代碼轉(zhuǎn)成匯編代碼,結(jié)果是 main.s。
  • 匯編代碼是介于高級(jí)語言和機(jī)器語言之間的一種語言,更接近機(jī)器。

5.3 匯編:把匯編轉(zhuǎn)成機(jī)器碼

gcc -c main.s -o main.o
  • -c:只執(zhí)行編譯到匯編的這一步,生成目標(biāo)文件(main.o)。
  • 目標(biāo)文件是二進(jìn)制的,但還不能直接運(yùn)行。

5.4 鏈接:生成可執(zhí)行文件

gcc main.o -o main
  • 鏈接器負(fù)責(zé)將目標(biāo)文件和系統(tǒng)庫一起鏈接,生成最終的可執(zhí)行文件。

各文件內(nèi)容對(duì)比:

文件名

內(nèi)容類型

cat/vim

查看結(jié)果

更合適的查看方式

main.i

預(yù)處理后的源碼

源碼,可讀

無需額外工具,直接查看

main.s

匯編代碼

匯編指令,可讀

無需額外工具,直接查看

main.o

二進(jìn)制目標(biāo)文件

亂碼,不可讀

objdumpreadelf

main

可執(zhí)行文件,機(jī)器碼

亂碼,不可讀

objdumpreadelf

PS:   gcc 和 g++ 的用法及參數(shù)基本相同。要編譯 C++ 程序,只需把命令中的 gcc 換成 g++,比如編譯 main.cpp:g++ main.cpp -o main。C++ 程序的文件通常以 .cpp 為后綴,如 main.cpp。

六、調(diào)試?yán)鳎篏DB 上線了

寫代碼,最怕的是:程序掛了,但根本不知道為什么掛。這時(shí),調(diào)試工具 GDB 就派上用場(chǎng)了。

6.1 用 gcc 編譯時(shí)加調(diào)試信息

gcc -g main.c -o main
  • -g 選項(xiàng)主要是生成調(diào)試信息,方便用 GDB 調(diào)試。

6.2 常用 GDB 命令

  • 1. 啟動(dòng) GDB:
gdb ./main
  • 進(jìn)入 GDB 調(diào)試模式。

2. 設(shè)置斷點(diǎn):

break <行號(hào)>
  • 比如 break 10,在代碼第 10 行 設(shè)置斷點(diǎn) 。

3. 運(yùn)行程序:

run
  • 運(yùn)行程序,停在斷點(diǎn)處。

4. 單步執(zhí)行:

  • 單步執(zhí)行,不進(jìn)入函數(shù)
next
  • 單步執(zhí)行,進(jìn)入函數(shù)。
step

選擇 next 跳過函數(shù),step 進(jìn)入函數(shù),按需使用即可!

5. 查看變量值:

print <變量名>
  • 比如 print x,顯示當(dāng)前變量 x 的值。

6. 查看當(dāng)前代碼(上下文代碼)

  • 查看當(dāng)前執(zhí)行位置的代碼:
list

默認(rèn)顯示當(dāng)前斷點(diǎn)附近的代碼。

  • 指定顯示某行附近的代碼:
list <行號(hào)>

比如 list 20,顯示第 20 行附近的代碼。

7. 打印函數(shù)調(diào)用棧

  • 查看調(diào)用棧信息:
backtrace
  • 顯示當(dāng)前函數(shù)被哪個(gè)函數(shù)調(diào)用,調(diào)用者又是誰,一層層往上追溯。
  • 對(duì)于分析崩潰點(diǎn)(coredump)特別有用。

8. 查看所有斷點(diǎn)

  • 列出當(dāng)前所有斷點(diǎn):
info breakpoints

可以看到每個(gè)斷點(diǎn)的編號(hào)、位置等信息。

9. 刪除斷點(diǎn)

  • 刪除某個(gè)斷點(diǎn):
delete <斷點(diǎn)編號(hào)>

比如 delete 1,刪除編號(hào)為 1 的斷點(diǎn)。

  • 刪除所有斷點(diǎn):
delete

10. 繼續(xù)運(yùn)行程序

  • 從當(dāng)前斷點(diǎn)繼續(xù)運(yùn)行程序:
continue
  • 程序會(huì)從當(dāng)前斷點(diǎn)繼續(xù)跑,直到遇到下一個(gè)斷點(diǎn)或結(jié)束。

11. 退出調(diào)試:

quit

七、常見問題排查

7.1 缺少頭文件

報(bào)錯(cuò):stdio.h: No such file or directory

原因:簡(jiǎn)單說,編譯器找不到 stdio.h 這個(gè)標(biāo)準(zhǔn)頭文件,可能是系統(tǒng)里沒裝編譯工具包,缺了開發(fā)相關(guān)的庫。

解決方法:

  • 在 Ubuntu/Debian 系統(tǒng)上,安裝必備工具包:使用命令 sudo apt install build-essential    這會(huì)把 gcc、g++ 和相關(guān)頭文件都裝上。
  • 在 CentOS/Red Hat 系統(tǒng)上,安裝開發(fā)工具:sudo yum groupinstall "Development Tools"這樣能確保編譯環(huán)境完整無缺。

7.2 “段錯(cuò)誤”

報(bào)錯(cuò):Segmentation fault (core dumped)

原因:簡(jiǎn)單說,程序想訪問一塊不該碰的內(nèi)存,比如:

  • 用了“野指針”(指針沒初始化,隨便指向了某個(gè)未知地址)。
  • 數(shù)組越界了,訪問了數(shù)組的第“10086”個(gè)元素,而數(shù)組長(zhǎng)度只有 100 個(gè)。
  • 使用了已經(jīng)釋放的內(nèi)存。

解決方法:

1、檢查指針和數(shù)組:

  • 確保指針初始化,比如:
int *ptr ; 
*ptr = 10;  // 未初始化就賦值,肯定會(huì)報(bào)錯(cuò):Segmentation fault (core dumped)
  • 不要訪問超出數(shù)組范圍的元素,比如訪問 arr[10086],而數(shù)組只有 100 個(gè)元素。
  • 如果是動(dòng)態(tài)內(nèi)存分配(malloc/free),檢查是否釋放了兩次。

2、用 GDB 調(diào)試:

  • 編譯時(shí)加上 -g 參數(shù)生成調(diào)試信息:
gcc -g main.c -o main
  • 然后用 GDB 跑程序:
gdb ./main
run
  • 程序崩潰時(shí),輸入:
bt

GDB 會(huì)告訴你出錯(cuò)在哪一行。

調(diào)試時(shí)別慌,找到那行代碼,慢慢改,段錯(cuò)誤就能搞定!

7.3 鏈接錯(cuò)誤

報(bào)錯(cuò):undefined reference to '某函數(shù)'

原因:這句話的意思很簡(jiǎn)單,你的代碼用到了一個(gè)函數(shù),但是編譯器在鏈接階段找不到它的實(shí)現(xiàn)。可能是:

  • 忘了加實(shí)現(xiàn)的文件:函數(shù)寫在另一個(gè) .c 文件里,編譯時(shí)沒有包含進(jìn)去。
  • 漏了庫的鏈接:用到了外部庫的函數(shù),但沒告訴編譯器要用哪個(gè)庫。
  • 函數(shù)聲明沒問題,函數(shù)實(shí)現(xiàn)卻沒寫,編譯器不知道該去哪里找它。

解決方法:

  • 如果函數(shù)是你自己寫的,確保編譯時(shí)包含了所有相關(guān)文件:
gcc main.c func.c -o main

沒加就補(bǔ)上!

  • 如果是庫函數(shù),比如用到了數(shù)學(xué)庫的 sqrt,需要鏈接對(duì)應(yīng)的庫,加上 -lm:
gcc main.c -lm -o main

這里的 -lm 表示鏈接數(shù)學(xué)庫(math library)。

  • 檢查函數(shù)實(shí)現(xiàn)是否真的寫了!如果只是聲明了函數(shù):
void my_function();

但實(shí)現(xiàn)忘了寫,肯定會(huì)報(bào)錯(cuò)。趕緊補(bǔ)上實(shí)現(xiàn)!

這個(gè)鏈接錯(cuò)誤其實(shí)很常見,仔細(xì)檢查文件和庫就能解決!

7.4 未定義的函數(shù)引用

報(bào)錯(cuò):undefined reference to '某函數(shù)'

原因:簡(jiǎn)單說,這個(gè)報(bào)錯(cuò)就是你的代碼用到了某個(gè)函數(shù),但編譯器找不到它的實(shí)現(xiàn)??赡苁悄銢]把實(shí)現(xiàn)的文件加到編譯命令里,或者用到了外部庫卻忘了鏈接。

解決方法:

1、如果是你自己寫的函數(shù),確認(rèn)它的實(shí)現(xiàn)文件是否加到編譯命令里,比如:

gcc main.c func.c -o main

沒加就補(bǔ)上!

2、如果是用的庫函數(shù),比如數(shù)學(xué)庫里的 sqrt,就加上對(duì)應(yīng)的庫鏈接,比如:

gcc main.c -lm -o main

這個(gè) -lm 是告訴編譯器“我要用數(shù)學(xué)庫”。

7.5 GDB 提示沒有調(diào)試信息

報(bào)錯(cuò):No debugging symbols found

原因:這個(gè)報(bào)錯(cuò)意思很簡(jiǎn)單:你的程序沒帶“調(diào)試信息”。GDB 說,“你讓我調(diào)試代碼,可你不給我地圖(調(diào)試信息),我咋知道問題在哪?”

解決方法:

編譯的時(shí)候記得加上 -g 參數(shù),這是讓 GCC 幫你把調(diào)試信息打包進(jìn)去,比如:

gcc -g main.c -o main

這個(gè) -g 就是那個(gè)“地圖”。

然后再用 GDB 調(diào)試:

gdb ./main

這下 GDB 才知道代碼怎么走的,可以幫你查問題了!

八、總結(jié):gcc/g++/gdb 不是魔法,用熟了像開掛

gcc/g++ 就像開發(fā)中的“瑞士軍刀”,功能全面卻不復(fù)雜,用熟了它們,開發(fā)效率會(huì)直線上升:

  • 從編譯到優(yōu)化,輕松搞定,讓程序又快又穩(wěn)。
  • 從調(diào)試到排錯(cuò),有了 gdb,分析問題更加清晰直觀。
  • 掌握了 gcc/g++ 和 gdb,任何 C/C++ 程序編譯調(diào)試都不在話下!

看完這篇文章,你是不是覺得 gcc/g++ 和 gdb 的用法更清晰了?趕緊上手操作一下吧,實(shí)踐是掌握工具的最快捷徑!

不過,項(xiàng)目復(fù)雜了,總手動(dòng)敲一長(zhǎng)串的 gcc 命令是不是太麻煩了?

責(zé)任編輯:武曉燕 來源: 跟著小康學(xué)編程
相關(guān)推薦

2021-10-09 15:49:00

5G網(wǎng)絡(luò)技術(shù)

2016-03-01 14:51:18

云計(jì)算DevOps

2022-03-22 07:37:04

FeignSpringRibbon

2016-01-08 11:00:14

OpenStack云計(jì)算

2019-02-20 14:35:57

區(qū)塊鏈數(shù)字貨幣比特幣

2023-07-10 09:38:06

兼容性測(cè)試方案

2021-11-25 07:43:56

CIOIT董事會(huì)

2017-08-02 10:43:39

深度學(xué)習(xí)TensorFlowRNN

2021-11-10 16:03:42

Pyecharts Python可視化

2018-10-29 15:20:03

2020-07-05 09:17:20

云桌面

2020-06-04 15:16:46

云計(jì)算

2021-06-07 10:05:56

性能優(yōu)化Kafka

2022-06-22 09:06:54

CSS垂直居中代碼

2019-03-17 16:48:51

物聯(lián)網(wǎng)云計(jì)算數(shù)據(jù)信息

2018-04-02 10:47:55

程序員AI打開方式

2021-01-11 10:47:09

IT部門網(wǎng)絡(luò)管理

2022-08-16 08:33:06

DevOps實(shí)踐

2018-07-03 09:41:23

數(shù)據(jù)庫系統(tǒng) 計(jì)算機(jī)

2024-08-20 15:56:54

點(diǎn)贊
收藏

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