沒有調試器的情況下編寫程序時最糟糕的狀況是什麼?編譯時跪着祈禱不要出錯?用血祭召喚惡魔幫你運行程序?或者在每一行代碼間添加printf("test")語句來定位錯誤點?如你所知,編寫程序時不使用調試器的話是不方便的。幸好,linux下調試還是很方便的。大多數人使用的IDE都集成了調試器,但 linux 最著名的調試器是命令行形式的C/C++調試器GDB。然而,與其他命令行工具一致,DGB需要一定的練習才能完全掌握。這裏,我會告訴你GDB的基本情況及使用方法。
安裝GDB
大多數的發行版倉庫中都有GDB
Debian 或 Ubuntu
- $ sudo apt-get install gdb
Arch Linux
- $ sudo pacman -S gdb
Fedora,CentOS 或 RHEL:
- $sudo yum install gdb
如果在倉庫中找不到的話,可以從官網中下載。
示例代碼
當學習GDB時,最好有一份代碼,動手試驗。下列代碼是我編寫的簡單例子,它可以很好的體現GDB的特性。將它拷貝下來並且進行實驗——這是最好的方法。
- #include <stdio.h>
- #include <stdlib.h>
- int main(int argc, char **argv)
- {
- int i;
- int a=0, b=0, c=0;
- double d;
- for (i=0; i<100; i++)
- {
- a++;
- if (i>97)
- d = i / 2.0;
- b++;
- }
- return 0;
- }
GDB的使用
首先最重要的,你需要使用編譯器的 “-g“選項來編譯程序,這樣可執行程序才能通過GDB來運行。通過下列語句開始調試:
- $ gdb -tui [可執行程序名]
使用”-tui“選項可以將代碼顯示在一個漂亮的交互式窗口內(所以被稱爲“文本用戶界面 TUI”),在這個窗口內可以使用光標來操控,同時在下面的GDB shell中輸入命令。
現在我們可以在程序的任何地方設置斷點。你可以通過下列命令來爲當前源文件的某一行設置斷點。
- break [行號]
或者爲一個特定的函數設置斷點:
- break [函數名]
甚至可以設置條件斷點
- break [行號] if [條件]
例如,在我們的示例代碼中,可以設置如下:
- break 11 if i > 97
這樣,程序循環97次之後停留在“a++”語句上。這樣是非常方便的,避免了我們需要手動循環97次。
最後但也是很重要的是,我們可以設置一個“觀察斷點”,當這個被觀察的變量發生變化時,程序會被停止。
- watch [變量]
這裏我們可以設置如下:
- watch d
當d的值發生變化時程序會停止運行(例如,當i>97爲真時)。
當設置斷點後,使用"run"命令開始運行程序,或按如下所示:
- r [程序的輸入參數(如果有的話)]
gdb中,大多數的命令單詞都可以簡寫爲一個字母。
不出意外,程序會停留在11行。這裏,我們可以做些有趣的事情。下列命令:
- bt
回溯功能(backtrace)可以讓我們知道程序如何到達這條語句的。
- info locals
這條語句會顯示所有的局部變量以及它們的值(你可以看到,我沒有爲d設置初始值,所以它現在的值是任意值)。
當然:
- p [變量]
這個命令可以顯示特定變量的值,而更進一步:
- ptype [變量]
可以顯示變量的類型。所以這裏可以確定d是double型。
既然已經到這一步了,我麼不妨這麼做:
- set var [變量] = [新的值]
這樣會覆蓋變量的值。不過需要注意,你不能創建一個新的變量或改變變量的類型。我們可以這樣做:
- set var a = 0
如其他優秀的調試器一樣,我們可以單步調試:
- step
使用如上命令,運行到下一條語句,有可能進入到一個函數裏面。或者使用:
- next
這可以直接運行下一條語句,而不進入子函數內部。
結束測試後,刪除斷點:
- delete [行號]
從當前斷點繼續運行程序:
- continue
退出GDB:
- quit
總之,有了GDB,編譯時不用祈禱上帝了,運行時不用血祭了,再也不用printf(“test“)了。當然,這裏所講的並不完整,而且GDB的功能遠遠不止於此。所以我強烈建議你自己更加深入的學習它。我現在感興趣的是將GDB整合到Vim中。同時,這裏有一個備忘錄記錄了GDB所有的命令行,以供查閱。
via: http://xmodulo.com/gdb-command-line-debugger.html
作者:Adrien Brochard 譯者:SPccman 校對:wxy
gdb斷點(在gdb中可以配合shell命令)—斷點設置 斷點管理—單文件,多文件(c和c++)
在gdb中使用shell命令的方式是:
(gdb) shell command
但是cd是內置命令,就不要使用shell了。
gdb原文件的搜索
search text: 該命令可顯示在當前文件中包含text串的下一行。
Reverse-search text: 該命令可以顯示包含text 的前一行。
斷點設置:
單文件設置斷點:
break命令(可以簡寫爲b)可以用來在調試的程序中設置斷點,該命令有如下四種形式:
(gdb) break line-number 使程序恰好在執行給定行之前停止。
(gdb) break function-name 使程序恰好在進入指定的函數之前停止。
(gdb) break line-or-function if condition 如果condition(條件)是真,程序到達指定行或函數時停止。
(gdb) break routine-name 在指定例程的入口處設置斷點
C多文件設置斷點:
如果該程序是由很多原文件構成的,你可以在各個原文件中設置斷點,而不是在當前的原文件中設置斷點,其方法如下:
(gdb) break filename:line-number 多文件中,在指定文件::指定行,設置斷點
(gdb) break filename:function-name 多文件中,在指定文件::指定函數處,設置斷點 (在C語言中,不存在域名的問題)
要想設置一個條件斷點,可以利用break if命令,如下所示:
(gdb) break line-or-function if expr 設置條件斷點
例:
(gdb) break 46 if testsize==100
斷點繼續運行:countinue 命令
C++多文件設置斷點:
錯誤跨文件設置斷點方法:
1. b str_parser.cpp:StrParser::LocalSegment //其中StrParser是域名
2. b str_parser.cpp:LocalSegment
3. b str_parser.cpp:StrParser::LocalSegment()
正確跨文件設置方法:
1. b StrParser::LocalSegment //其中StrParser是域名
2. b str_parser.cpp:1361 (StrParser::LocalSegment所在行)
斷點管理:
顯示當前gdb的斷點信息:
(gdb) info break
他會以如下的形式顯示所有的斷點信息:
Num Type Disp Enb Address What
1 breakpoint keep y 0x000028bc in init_random at qsort2.c:155
2 breakpoint keep y 0x0000291c in init_organ at qsort2.c:168
刪除指定的某個斷點:
(gdb) delete breakpoint 1
該命令將會刪除編號爲1的斷點,如果不帶編號參數,將刪除所有的斷點
(gdb) delete breakpoint
禁止使用某個斷點
(gdb) disable breakpoint 1
該命令將禁止斷點 1,同時斷點信息的 (Enb)域將變爲 n
允許使用某個斷點
(gdb) enable breakpoint 1
該命令將允許斷點 1,同時斷點信息的 (Enb)域將變爲 y
清除原文件中某一代碼行上的所有斷點
(gdb)clean number
注:number 爲原文件的某個代碼行的行號