日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Ubuntu >内容正文

Ubuntu

GDB调试程序-Ubuntu

發布時間:2024/4/15 Ubuntu 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 GDB调试程序-Ubuntu 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

導讀:


用GDB調試程序
出自Ubuntu中文
目錄
[隱藏]
1 GDB概述
2 一個調試示例
3 使用GDB
4 GDB的命令概貌
5 GDB中運行UNIX的shell程序
6 在GDB中運行程序
7 調試已運行的程序
8 暫停/恢復程序運行
8.1 設置斷點(Break Points)
8.2 設置觀察點(WatchPoint)
8.3 設置捕捉點(CatchPoint)
8.4 維護停止點
8.5 停止條件維護
8.6 為停止點設定運行命令
8.7 斷點菜單
8.8 恢復程序運行和單步調試
8.9 信號(Signals)
8.10 線程(Thread Stops)
9 查看棧信息
10 查看源程序
10.1 顯示源代碼
10.2 搜索源代碼
10.3 指定源文件的路徑
10.4 源代碼的內存
11 查看運行時數據
11.1 表達式
11.2 程序變量
11.3 數組
11.4 輸出格式
11.5 查看內存
11.6 自動顯示
11.7 設置顯示選項
11.8 歷史記錄
11.9 GDB環境變量
11.10 查看寄存器
12 改變程序的執行
12.1 修改變量值
12.2 跳轉執行
12.3 產生信號量
12.4 強制函數返回
12.5 強制調用函數
13 在不同語言中使用GDB
14 後記
15 相關詞條
?
?
用GDB調試程序(zz)
作 者:haoel (QQ是:753640,MSN是:haoel@hotmail.com )
來源:http://blog.csdn.net/haoel/archive/2003/07/02/2879.aspx
1.GDB概述
GDB 是GNU開源組織發布的一個強大的UNIX
下的程序調試工具。或許,各位比較喜歡那種圖形界面方 式的,像VC、BCB等IDE
的調試,但如果你是在 UNIX平臺下做軟件,你會發現GDB這個調試工具有比VC、BCB
的圖形化調試器 更強大的功能。所謂“寸有所長,尺有所短”就是這個道理。

一般來說,GDB主要幫忙你完成下面四個方面的功能:

啟動你的程序,可以按照你的自定義的要求隨心所欲的運行程序。
可讓被調試的程序在你所指定的調置的斷點處停住。(斷點可以是條件表達式)
當程序被停住時,可以檢查此時你的程序中所發生的事。
動態的改變你程序的執行環境。
從上面看來,GDB
和一般的調 試工具沒有什么兩樣,基本上也是完成這些功能,不過在細節上,你會發現
GDB
這個調試工具的強大,大家可能比較習慣了圖形化的調試工具, 但有時候,命令行的調試?
ぞ呷從兇磐夾位ぞ咚荒芡瓿傻墓δ堋H夢頤且灰豢蠢礎?

2.一個調試示例
源程序:tst.c

?1 #include <stdio.h
>????????????????????????????????????????????????????????????????
?2
?3 int func(int n)
?4 {
?5???? int sum=0,i;
?6???? for(i=1; i<=n; i++)
?7???? {
?8???????? sum+=i;
?9???? }
10???? return sum;
11 }
12
13
14 int main()
15 {
16???? int i;
17???? long result = 0;
18???? for(i=1; i<=100; i++)
19???? {
20???????? result += i;
21???? }
22
23???? printf("result[1-100] = %ld /n", result );
24???? printf("result[1-250] = %d /n", func(250) );
25???? return 0;
26 }
27

編譯生成執行文件:

$gcc -g -Wall tst.c -o tst

使用GDB調試:

$ gdb tst?? <---------- 啟動GDB
GNU gdb 6.7.1-debian
Copyright (C) 2007 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html >
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.? Type "show copying"
and "show warranty" for details.
This GDB was configured as "i486-linux-gnu"...
Using host libthread_db library "/lib/tls/i686/cmov/libthread_db.so.1".
(gdb) l 1?? <-------------------- l命令相當于list,從第一行開始例出原碼。
1?????? #include <stdio.h>
2
3?????? int func(int n)
4?????? {
5?????????????? int sum=0,i;
6?????????????? for(i=1; i<=n; i++)
7?????????????? {
8?????????????????????? sum+=i;
9?????????????? }
10????????????? return sum;
(gdb)??? <-------------------- 直接回車表示,重復上一次命令
11????? }
12
13
14????? int main()
15????? {
16????????????? int i;
17????????????? long result = 0;
18????????????? for(i=1; i<=100; i++)
19????????????? {
20???result += i;
(gdb) break 16??? <-------------------- 設置斷點,在源程序第16行處。
Breakpoint 1 at 0x80483b2: file tst.c, line 16.
(gdb) break func?? <-------------------- 設置斷點,在函數func()入口處。
Breakpoint 2 at 0x804837a: file tst.c, line 5.
(gdb) info break?? <-------------------- 查看斷點信息。
Num Type?????????? Disp Enb Address??? What
1?? breakpoint???? keep y?? 0x080483b2 in main at tst.c:16
2?? breakpoint???? keep y?? 0x0804837a in func at tst.c:5
(gdb) r??? <--------------------- 運行程序,run命令簡寫
Starting program: /home/dbzhang/tst

Breakpoint 1, main () at tst.c:17
17??long result = 0;
(gdb) n???? <--------------------- 單條語句執行,next命令簡寫。
18??for(i=1; i<=100; i++)
(gdb) n
20???result += i;
(gdb) n
18??for(i=1; i<=100; i++)
(gdb) n
20???result += i;
(gdb) c??? <--------------------- 繼續運行程序,continue命令簡寫。
Continuing.
result[1-100] = 5050??? <----------程序輸出。

Breakpoint 2, func (n=250) at tst.c:5
5??int sum=0,i;
(gdb) n
6??for(i=1; i<=n; i++)
(gdb) p i?? <--------------------- 打印變量i的值,print命令簡寫。
$1 = -1074568236
(gdb) n
8???sum+=i;
(gdb) n
6??for(i=1; i<=n; i++)
(gdb) p sum
$2 = 1
(gdb) n
8???sum+=i;
(gdb) p i
$3 = 2
(gdb) n
6??for(i=1; i<=n; i++)
(gdb) p sum
$4 = 3
(gdb) bt?? <--------------------- 查看函數堆棧。
#0? func (n=250) at tst.c:6
#1? 0x080483f1 in main () at tst.c:24
(gdb) finish?? <--------------------- 退出函數。
Run till exit from #0? func (n=250) at tst.c:6
0x080483f1 in main () at tst.c:24
24??printf("result[1-250] = %d /n", func(250) );
Value returned is $5 = 31375
(gdb) c?? <--------------------- 繼續運行。
Continuing.
result[1-250] = 31375??? <----------程序輸出。

Program exited normally.?? <--------程序退出,調試結束。
(gdb) q?? <--------------------- 退出gdb。


好了,有了以上的感性認識,還是讓我們來系統地認識一下gdb吧。

3.使用GDB
一般來說GDB主要調試的是C/C++的程序。要調試C/C
++的程序,首先在編譯時,我們必須要把調試信息 加到可執行文件中。使用編譯器(cc/
gcc/g++)的 -g 參數可以做到這一點。如:

$gcc -g -Wall hello.c -o hello
$g++ -g -Wall hello.cpp -o hello

如果沒有-g
,你將看不見程序的函數名、變量名,所代替的全是運行時的內存地址。當你用-g
把調試信息加入之後,并成功編譯目標代 碼以後,讓我們來看看如何用gdb來調試他。

啟動GDB的方法有以下幾種:

gdb <program>
program也就是你的執行文件,一般在當然目錄下。
gdb <program> core
用gdb同時調試一個運行程序和core文件,core是程序非法執行後core dump
後 產生的文件。
gdb <program> <PID>
如果你的程序是一個服務程序,那么你可以指定這個服務 程序運行時的進程ID。gdb
會自動attach上去,并調試他。program應該在PATH環境變量中搜索得到。
GDB啟動時,可 以加上一些GDB的啟動開關,詳細的開關可以用gdb -help
查看。我在下面只例舉一些比較常用的參數:

-symbols <file>
-s <file>
從指定文件中讀取符號表。
-se file
從指定文件中讀取符號表信息,并把他用在可執行文件中。
-core <file>
-c <file>
調試時core dump的core文件。
-directory <directory>
-d <directory>
加入一個源文件的搜索路徑。默認搜索路徑是環境變量中PATH所定義的路徑。
4. GDB的命令概貌
啟動gdb後,就你被帶入gdb的調試環境中,就可以使用gdb的命令開始調試程序了,gdb
的命令可以使用help命 令來查看,如下所示:

$ gdb
GNU gdb 6.7.1-debian
Copyright (C) 2007 Free Software Foundation, Inc.
License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html >
This is free software: you are free to change and redistribute it.
There is NO WARRANTY, to the extent permitted by law.? Type "show copying"
and "show warranty" for details.
This GDB was configured as "i486-linux-gnu".
(gdb) help
List of classes of commands:

aliases -- Aliases of other commands
breakpoints -- Making program stop at certain points
data -- Examining data
files -- Specifying and examining files
internals -- Maintenance commands
obscure -- Obscure features
running -- Running the program
stack -- Examining the stack
status -- Status inquiries
support -- Support facilities
tracepoints -- Tracing of program execution without stopping the program
user-defined -- User-defined commands

Type "help" followed by a class name for a list of commands in that class.
Type "help all" for the list of all commands.
Type "help" followed by command name for full documentation.
Type "apropos word" to search for commands related to "word".
Command name abbreviations are allowed if unambiguous.
(gdb)

gdb 的命令很多,gdb把之分成許多個種類。help命令只是例出gdb
的命令種類,如果要看種類中的命令,可以使用help <class> 命令,如:help
breakpoints,查看設置斷點的所有命令。也可以直接help <command
>來查看命令的幫助。

gdb
中,輸入命令時,可以不用打全命令,只用打命令的前幾個字符就可以了,當然,命令的?
凹父鱟址Ω靡曛咀乓桓鑫ㄒ壞拿睿 贚inux下,你可以敲擊兩次TAB
鍵來補齊命令的全稱,如果有重復的,那么gdb會把其例出來。

示例一:在進入函數func時,設置一個斷點。可以敲入break func,或是直接就是b func

(gdb) b func
Breakpoint 1 at 0x804837a: file tst.c, line 5.

示例二:敲入b按兩次TAB鍵,你會看到所有b打頭的命令:

(gdb) b
backtrace break bt
(gdb)

示例三:只記得函數的前綴,可以這樣:

(gdb) b make_ <按TAB鍵>
(再按下一次TAB鍵,你會看到:)
make_a_section_from_file make_environ
make_abs_section make_function_type
make_blockvector make_pointer_type
make_cleanup make_reference_type
make_command make_symbol_completion_list
(gdb) b make_
GDB把所有make開頭的函數全部例出來給你查 看。

示例四:調試C++的程序時,有可以函數名一樣。如:

(gdb) b 'bubble( M-?
bubble(double,double) bubble(int,int)
(gdb) b 'bubble(

你可以查看到C++中的所有的重載函數及參數。(注:M-?和“按兩次TAB
鍵”是一個意思)

要退出gdb時,只用發quit或命令簡稱q就行了。

5. GDB中運行UNIX的shell程序
在gdb環境中,你可以執行UNIX的shell的命令,使用gdb的shell命令來完 成:

shell <command string>

調用UNIX的shell來執行<command string>,環境變量SHELL中定義的UNIX的shell
將會被用 來執行<command string>,如果SHELL沒有定義,那就使用UNIX的標準shell:/
bin/sh。(在 Windows中使用Command.com或cmd.exe)

還有一個gdb命令是make:

make <make-args>

可以在gdb中執行make命令來重新build自己的程序。這個命令等價于“shell make <make
-args>”。

?

6. 在GDB中運行程序
當以gdb <program>方式啟動gdb後,gdb會在PATH路徑和當前目錄中搜索<program
>的源文件。如要確認 gdb是否讀到源文件,可使用l或list命令,看看gdb
是否能列出源代碼。

在gdb中,運行程序使用r或是run
命令。程序的運行,你有可能需要設置下面四方面的事。

1、程序運行參數。
set args 可指定運行時參數。(如:set args 10 20 30 40 50)
show args 命令可以查看設置好的運行參數。

2、運行環境。
path <dir> 可設定程序的運行路徑。
show paths 查看程序的運行路徑。
set environment varname [=value] 設置環境變量。如:set env USER=hchen
show environment [varname] 查看環境變量。

3、工作目錄。
cd <dir> 相當于shell的cd命令。
pwd 顯示當前的所在目錄。

4、程序的輸入輸出。
info terminal 顯示你程序用到的終端的模式。
使用重定向控制程序輸出。如:run > outfile
tty命令可以指寫輸入輸出的終端設備。如:tty /dev/ttyb

?

7.調試已運行的程序
兩種方法:

在UNIX下用ps查看正在運行的程序的PID(進程ID),然後用gdb <program> PID
格式掛接正在運行的程 序。
先用gdb <program>關聯上源代碼,并進行gdb,在gdb中用attach命令來掛接進程的PID
。并用 detach來取消掛接的進程。
8. 暫停/恢復程序運行
調試程序中,暫停程序運行是必須的,GDB
可以方便地暫停程序的 運行。你可以設置程序的在哪行停住,在什么條件下停住,在收到?
裁蔥藕攀蓖M鵲取R員閿諛悴榭叢誦惺鋇謀淞浚約霸誦惺鋇牧鞒獺?

當進程被gdb停住時,你可以使用info program
來查看程序的是否在運行,進程號,被暫停的原因。

在gdb中,我們可以有以下幾種暫停方式:斷點(BreakPoint)、觀察點(Watch Point
)、捕捉點(Catch Point)、信號(Signals)、線程停止(Thread Stops
)。如果要恢復程序運行,可以使用c或是 continue命令。

8.1 設置斷點(Break Points)
我們用break命令來設置斷點。正面有幾點設置斷點的方法:

break <function>
在進入指定函數時停住。C++中可以使用class::function或 function(type,type)
格式來指定函數名。
break <linenum>
在指定行號停住。
break +offset
break -offset
在當前行號的前面或後面的offset行停住。offiset為自然數。
break filename:linenum
在源文件filename的linenum行處停住。
break filename:function
在源文件filename的function函數的入口處停住。
break *address
在 程序運行的內存地址處停住。
break
break命令沒有參數時,表示在下一條指令處停住。
break ... if <condition>
...可以是上述的參數,condition
表示條件,在條件成立時停住。比如在循環境體中,可以 設置break if i=100,表示當i
為100時停住程序。
查看斷點時,可使用info命令,如下所示:(注:n表示斷點號)

info breakpoints [n]
info break [n]
8.2 設置觀察點(WatchPoint)
觀 察點一般來觀察某個表達式(變量也是一種表達式)的值是否有變化了,如果有變化,?
砩賢W〕絳頡N頤怯邢旅嫻募鋼址椒ɡ瓷柚霉鄄斕悖?

watch <expr>
為表達式(變量)expr設置一個觀察點。一量表達式值有變化時,馬上停住程序。
rwatch <expr>
當表達式(變量)expr被讀時,停住程序。
awatch <expr>
當表達式 (變量)的值被讀或被寫時,停住程序。
info watchpoints
列出當前所設置了的所有觀察點。
8.3 設置捕捉點(CatchPoint)
?? 你可設置捕捉點來補捉程序運行時的一些事件。如:載入共享庫(動態鏈接庫)或是C
++的異常。 設置捕捉點的格式為:

catch <event>
當event發生時,停住程序。event可以是下面的內容:

throw 一個C++拋出的異常。(throw為關鍵字)
catch 一個C++捕捉到的異常。(catch為關鍵字)
exec 調用系統調用exec時。(exec為關鍵字,目前此功能只在HP-UX下有用)
fork 調用系統調用fork時。(fork為關鍵字,目前此功能只在HP-UX下有用)
vfork 調用系統調用vfork時。(vfork為關鍵字,目前此功能只在HP-UX下有用)
load 或 load <libname> 載入共享庫(動態鏈接庫)時。(load
為關鍵字,目前此功能只在HP-UX下有用)
unload 或 unload <libname> 卸載共享庫(動態鏈接庫)時。(unload
為關鍵字,目前此功能只在HP-UX下有用)
tcatch <event>
只設置一次捕捉點,當程序停住以後,應點被自動刪除。

8.4 維護停止點
上面說了如何設置程序的停止點,GDB中的停止點也就是上述的三類。在GDB
中,如果你覺得已定義好的停止點 沒有用了,你可以使用delete、clear、disable、
enable這幾個命令來進行維護。

clear
清除所有的已定義的停止點。
clear <function>
clear <filename:function>
清除所有設置在函數上的停止點。
clear <linenum>
clear <filename:linenum>
清除所有設置在指定行上的停止點。
delete [breakpoints] [range...]
刪除指定的斷點,breakpoints為斷點號。如果不指定斷點號,則表示刪除所有的斷點。
range 表示斷點號的范圍(如:3-7)。其簡寫命令為d。

比刪除更好的一種方法是disable停止點,disable了的停止點,GDB
不會刪除,當你還需要時,enable即可,就好像回收站 一樣。

disable [breakpoints] [range...]
disable所指定的停止點,breakpoints為停止點號。 如果什么都不指定,表示disable
所有的停止點。簡寫命令是dis.
enable [breakpoints] [range...]
enable所指定的停止點,breakpoints為停止點號。
enable [breakpoints] once range...
enable所指定的停止點一次,當程序停止後,該停止點馬上被GDB自動disable。
enable [breakpoints] delete range...
enable所指定的停止點一次,當程序停止後,該停止點馬上被GDB自動刪 除。
8.5停止條件維護
前面在說到設置斷點時,我們提到過可以設置一個條件,當條件成立時,程序自動停止,?
饈且桓齜淺G 看蟮墓δ埽飫錚蟻胱潘鄧嫡飧鎏跫南喙匚っ睢R話憷此擔?
點設置一個條件,我們使用if
關鍵詞,後面跟其斷點條件。并且, 條件設置好後,我們可以用condition
命令來修改斷點的條件。(只有break和watch命令支持if,catch目前暫不支持if)

condition <bnum> <expression>
修改斷點號為bnum的停止條件為 expression。
condition <bnum>
清除斷點號為bnum的停止條件。

還有一個比較特殊的維護命令ignore,你可以指定程序運行時,忽略停止條件幾次。

ignore <bnum> <count>
表示忽略斷點號為bnum的停止條件count次。
8.6為停止點設定運行命令
我們可以使用GDB提供的command
命令來設置停止點的運行命令。也就是說,當運行的程序在被停止住時, 我們可以讓其自?
誦幸恍┍鸕拿睿夂苡欣兇遠魘浴6曰贕DB
的自動化調試是一個強大的支持。

commands [bnum]
... command-list ...
end

為斷點號bnum指寫一個命令列表。當程序被該斷點停住時,gdb
會依次運行命令列表中的命令。例如:

break foo if x>0
commands
printf "x is %d/n",x
continue
end

斷點設置在函數foo中,斷點條件是x>0,如果程序被斷住後,也就是,一旦x的值在foo
函數中大于0,GDB會自動打印出x的 值,并繼續運行程序。

如果你要清除斷點上的命令序列,那么只要簡單的執行一下commands命令,并直接在打個
end就行了。

8.7 斷點菜單
在C++中,可能會重復出現同一個名字的函數若干次(函數重載),在這種情況下,
break <function>不能告訴GDB要停在哪個函數的入口。當然,你可以使用break <
function(type)> 也就是把函數的參數類型告訴GDB,以指定一個函數。否則的話,GDB
會給你列出一個斷點菜單供你選擇你所需要的斷點。你只要輸入你菜單列表中的 編號就可?
粵恕H紓?

(gdb) b String::after
[0] cancel
[1] all
[2] file:String.cc; line number:867
[3] file:String.cc; line number:860
[4] file:String.cc; line number:875
[5] file:String.cc; line number:853
[6] file:String.cc; line number:846
[7] file:String.cc; line number:735
> 2 4 6
Breakpoint 1 at 0xb26c: file String.cc, line 867.
Breakpoint 2 at 0xb344: file String.cc, line 875.
Breakpoint 3 at 0xafcc: file String.cc, line 846.
Multiple breakpoints were set.
Use the "delete" command to delete unwanted
breakpoints.
(gdb)

可見,GDB列出了所有after的重載函數,你可以選一下列表編號就行了。0
表示放棄設置斷點,1表示所有函數都設置斷點。

8.8 恢復程序運行和單步調試
當程序被停住了,你可以用continue
命令恢復程序的運行直到程序結束,或下一個斷點到來。 也可以使用step或next
命令單步跟蹤程序。

continue [ignore-count]
c [ignore-count]
fg [ignore-count]
恢 復程序運行,直到程序結束,或是下一個斷點到來。ignore-count
表示忽略其後的斷點次數。continue,c,fg三個命令都是一樣 的意思。
step <count>
單步跟蹤,如果有函數調用,他會進入該函數。進入函數的前提是,此函數被編譯有
debug 信息。很像VC等工具中的step in。後面可以加count
也可以不加,不加表示一條條地執行,加表示執行後面的count條指令,然後再停 住。
next <count>
同樣單步跟蹤,如果有函數調用,他不會進入該函數。很像VC等工具中的step over
。 後面可以加count也可以不加,不加表示一條條地執行,加表示執行後面的count
條指令,然後再停住。
set step-mode
set step-mode on
打開step-mode模式,于是,在進行單步跟蹤時,程序不會因為沒有debug
信息而不停住。這個參數 有很利于查看機器碼。
set step-mod off
關閉step-mode模式。
finish
運行程序,直 到當前函數完成返回。并打印函數返回時的堆棧地址和返回值及參數值等信?
ⅰ?
until 或 u
當你厭倦了在一個循環體內單步 跟蹤時,這個命令可以運行程序直到退出循環體。
stepi 或 si
nexti 或 ni
單步跟蹤一條機器指令!一條程序代 碼有可能由數條機器指令完成,stepi和nexti
可以單步執行機器指令。與之一樣有相同功能的命令是 “display/i $pc”
, 當運行完這個命令後,單步跟蹤會在打出程序代碼的同時打出機器指令(也就是匯編代?
耄?
8.9信號(Signals)
信號是 一種軟中斷,是一種處理異步事件的方法。一般來說,操作系統都支持許多信號。?
繞涫?UNIX,比較重要應用程序一般都會處理信號。UNIX定義 了許多信號,比如SIGINT
表示中斷字符信號,也就是Ctrl+C的信號,SIGBUS表示硬件故障的信號;SIGCHLD
表示子進程 狀態改變信號;SIGKILL表示終止程序運行的信號,等等。信號量編程是UNIX
下非常重要的一種技術。

GDB有能力在你調試程序的時候處理任何一種信號,你可以告訴GDB
需要處理哪一種信號。你可以要求GDB
收到你所指定的信號時, 馬上停住正在運行的程序,以供你進行調試。你可以用GDB的
handle命令來完成這一功能。

handle <signal> <keywords...>

在GDB中定義一個信號處理。信號<signal>可以以SIG開頭或不以SIG
開頭,可以用定義一個要處理信號的范圍 (如:SIGIO- SIGKILL,表示處理從SIGIO
信號到SIGKILL的信號,其中包括SIGIO,SIGIOT,SIGKILL
三 個信號),也可以使用關鍵字 all
來標明要處理所有的信號。一旦被調試的程序接收到信號,運行程序馬上會被GDB
停住,以供調試。 其<keywords>可以是以下幾種關鍵字的一個或多個。

nostop
當被調試的程序收到信號時,GDB
不會停住程序的運行,但會打出消息告訴你收到這種信號。
stop
當 被調試的程序收到信號時,GDB會停住你的程序。
print
當被調試的程序收到信號時,GDB會顯示出一條信息。
noprint
當被調試的程序收到信號時,GDB不會告訴你收到信號的信息。
pass
noignore
當被調試的程序收到信號 時,GDB不處理信號。這表示,GDB
會把這個信號交給被調試程序會處理。
nopass
ignore
當被調試的程序 收到信號時,GDB不會讓被調試程序來處理這個信號。
info signals
info handle
查看有哪些信號在被 GDB檢測中。
8.10 線程(Thread Stops)
如果你程序是多線程的話,你可以定義你的斷點是否在所有的線程上,或是在某 個特定的?
叱獺DB很容易幫你完成這一工作。

break <linespec> thread <threadno>
break <linespec> thread <threadno> if ...

linespec指定了斷點設置在的源程序的行號。threadno指定了線程的ID,注意,這個ID是
GDB分配的,你可以通過 “info threads
”命令來查看正在運行程序中的線程信息。如果你不指定thread <threadno
>則表示 你的斷點設在所有線程上面。你還可以為某線程指定斷點條件。如:

(gdb) break frik.c:13 thread 28 if bartab > lim

當你的程序被GDB
停住時,所有的運行線程都會被停住。這方便你你查看運行程序的總體情況。而在你恢復?
絳蛟誦惺保械南叱桃不 岜換指叢誦小D橋率侵鶻淘詒壞ゲ降魘允薄?

9. 查看棧信息
當程序被停住了,你需要做的第一件事就是查看程序是在哪里停住的。當你的程序調用了?
桓齪牡刂罰 問詰木植勘淞慷薊岜謊谷搿罷弧保⊿tack
)中。你可以用GDB命令來查看當前的棧中的信息。

下面是一些查看函數調用棧信息的GDB命令:

backtrace
bt
打印當前的函數調用棧的所有信息。如:
(gdb) bt
#0? func (n=250) at tst.c:6
#1? 0x08048524 in main (argc=1, argv=0xbffff674) at tst.c:30
#2? 0x400409ed in __libc_start_main () from /lib/libc.so.6

從上可以看出函數的調用棧信息:__libc_start_main --> main() --> func()


backtrace <n>
bt <n>
n是一個正整數,表示只打印棧頂上n層的棧信息。
backtrace <-n>
bt <-n>
-n表一個負整數,表示只打印棧底下n層的棧信息。
如果你要查看某一層的 信息,你需要在切換當前的棧,一般來說,程序停止時,最頂層的?
瘓褪塹鼻罷唬綣鬩榭湊幌旅娌愕南晗感畔ⅲ紫紉齙氖喬謝壞鼻罷弧?

frame <n>
f <n>
n是一個從0開始的整數,是棧中的層編號。比如:frame 0,表示棧頂,frame 1
,表示棧的第二層。
up <n>
表示向棧的上面移動n層,可以不打n,表示向上移 動一層。
down <n>
表示向棧的下面移動n層,可以不打n,表示向下移動一層。

上面的命令,都會打印出移動到的棧層的信息。如果你不想讓其打出信息。你可以使用這?
雒睿?

select-frame <n> 對應于 frame 命令。
up-silently <n> 對應于 up 命令。
down-silently <n> 對應于 down 命令。

查看當前棧層的信息,你可以用以下GDB命令:

frame 或 f
會打印出這些信息:棧的層編號,當前的函數名,函數參數值,函數所在文件及行號,函?
蔥械降撓錁洹?
info frame
info f
這個命令會打印出更為詳細的當前棧層的信息,只不過,大多數都是運行時的內內地址。?
熱紓漢刂罰 饔煤牡刂罰壞饔煤牡刂罰殼暗暮怯墑裁囪某絳蠐镅?
寫成的、函數參數地址及值、局部變量的地址等等。如:
(gdb) info f
Stack level 0, frame at 0xbffff5d4:
eip = 0x804845d in func (tst.c:6); saved eip 0x8048524
called by frame at 0xbffff60c
source language c.
Arglist at 0xbffff5d4, args: n=250
Locals at 0xbffff5d4, Previous frame's sp is 0x0
Saved registers:
ebp at 0xbffff5d4, eip at 0xbffff5d8

info args
打印出當前函數的參數名及其值。
info locals
打印出當前函數中所有局部變量及其值。
info catch
打印出當前的函數中的異常處理信息。
10. 查看源程序
10.1 顯示源代碼
GDB 可以打印出所調試程序的源代碼,當然,在程序編譯時一定要加上-g
的參數,把源程序信息編譯到執行文件中。不然就看不到源程序了。當程序停下來以 後,
GDB會報告程序停在了那個文件的第幾行上。你可以用list
命令來打印程序的源代碼。還是來看一看查看源代碼的GDB命令吧。

list <linenum>
顯示程序第linenum行的周圍的源程序。
list <function>
顯示函數名為function的函數的源程序。
list
顯示當前行後面的源程序。
list -
顯示當前行前面的源程序。
一般是打印當前行的上5行和下5行,如果顯示函數是是上2行下8行,默認是10
行,當然,你也 可以定制顯示的范圍,使用下面命令可以設置一次顯示源程序的行數。

set listsize <count>
設置一次顯示源代碼的行數。
show listsize
查看 當前listsize的設置。
list命令還有下面的用法:

list <first>, <last>
顯示從first行到last行之間的源代碼。
list , <last>
顯示從當前行到last行之間的源代碼。
list +
往後顯示源代碼。
一般來說在 list後面可以跟以下這們的參數:

<linenum>?? 行號。
<+offset>?? 當前行號的正偏移量。
<-offset>?? 當前行號的負偏移量。
<filename:linenum>? 哪個文件的哪一行。
<function>? 函數名。
<filename:function> 哪個文件中的哪個函數。
<*address>? 程序運行時的語句在內存中的地址。

10.2 搜索源代碼
不僅如此,GDB還提供了源代碼搜索的命令:

forward-search <regexp>
search <regexp>
向前面搜索。
reverse-search <regexp>
全部搜索。
其中,<regexp
>就是正則表達式,也主一個字符串的匹配模式, 關于正則表達式,我就不在這里講了,還
請各位查看相關資料。

10.3 指定源文件的路徑
某些時候,用-g編譯過後的執行程序中只是包括了源文件的名字,沒有路徑名。GDB
提供了可以讓你指定 源文件的路徑的命令,以便GDB進行搜索。

directory <dirname ... >
dir <dirname ... >
加一個源文件 路徑到當前路徑的前面。如果你要指定多個路徑,UNIX
下你可以使用“:”,Windows下你可以使用“;”。
directory
清 除所有的自定義的源文件搜索路徑信息。
show directories
顯示定義了的源文件搜索路徑。
10.4源代碼的內存
你可以使用info line命令來查看源代碼在內存中的地址。info line
後面可以跟“行號”,“函數名”,“文件 名:行號”,“文件名:函數名”,這個命令會?
蠐〕鏊付ǖ腦綽朐讜誦惺鋇哪詿嫻刂罰紓?

(gdb) info line tst.c:func
Line 5 of "tst.c" starts at address 0x8048456 <func+6> and ends at 0x804845d <
func+13>.

還有一個命令(disassemble
)你可以查看源程序的當前執行時的機器碼,這個命令會把目前內存中的指令dump
出來。如下 面的示例表示查看函數func的匯編代碼。

(gdb) disassemble func
Dump of assembler code for function func:
0x8048450 <func>:?????? push?? %ebp
0x8048451 <func+1>:???? mov??? %esp,%ebp
0x8048453 <func+3>:???? sub??? $0x18,%esp
0x8048456 <func+6>:???? movl?? $0x0,0xfffffffc(%ebp)
0x804845d <func+13>:??? movl?? $0x1,0xfffffff8(%ebp)
0x8048464 <func+20>:??? mov??? 0xfffffff8(%ebp),%eax
0x8048467 <func+23>:??? cmp??? 0x8(%ebp),%eax
0x804846a <func+26>:??? jle??? 0x8048470 <func+32>
0x804846c <func+28>:??? jmp??? 0x8048480 <func+48>
0x804846e <func+30>:??? mov??? %esi,%esi
0x8048470 <func+32>:??? mov??? 0xfffffff8(%ebp),%eax
0x8048473 <func+35>:??? add??? %eax,0xfffffffc(%ebp)
0x8048476 <func+38>:??? incl?? 0xfffffff8(%ebp)
0x8048479 <func+41>:??? jmp??? 0x8048464 <func+20>
0x804847b <func+43>:??? nop
0x804847c <func+44>:??? lea??? 0x0(%esi,1),%esi
0x8048480 <func+48>:??? mov??? 0xfffffffc(%ebp),%edx
0x8048483 <func+51>:??? mov??? %edx,%eax
0x8048485 <func+53>:??? jmp??? 0x8048487 <func+55>
0x8048487 <func+55>:??? mov??? %ebp,%esp
0x8048489 <func+57>:??? pop??? %ebp
0x804848a <func+58>:??? ret
End of assembler dump.

11.查看運行時數據
在你調試程序時,當程序被停住時,你可以使用print命令(簡寫命令為p
),或是同義命令 inspect來查看當前程序的運行數據。print命令的格式是:

print <expr>
print /<f> <expr>

<expr>是表達式,是你所調試的程序的語言的表達式(GDB可以調試多種編程語言),<f
>是輸出的格式,比 如,如果要把表達式按16進制的格式輸出,那么就是/x。

11.1 表達式
print和許多GDB的命令一樣,可以接受一個表達式,GDB
會根據當前的程序運行的數據來計算這個表達式,既 然是表達式,那么就可以是當前程序?
誦兄械腸onst常量、變量、函數等內容。可惜的是GDB不能使用你在程序中所定義的宏。

表達式的語法應該是當前所調試的語言的語法,由于C/C
++是一種大眾型的語言,所以,本文中的例子都是關于C/C++的。(而關于用 GDB
調試其它語言的章節,我將在後面介紹)

在表達式中,有幾種GDB所支持的操作符,它們可以用在任何一種語言中。

@

是一個和數組有關的操作符,在後面會有更詳細的說明。
::

指定一個在文件或是一個函數中的變量。
{<type>} <addr>

表示一個指向內存地址<addr>的類型為type的一個對象。
11.2程序變量
在GDB中,你可以隨時查看以下 三種變量的值:

全局變量(所有文件可見的)
靜態全局變量(當前文件可見的)
局部變量(當前Scope可見的)
如果你的局部變量和全 局變量發生沖突(也就是重名),一般情況下是局部變量會隱藏全?
直淞浚簿褪撬擔綣桓鋈直淞亢鴕桓齪械木植勘淞客保綣鼻巴V溝 閽?
函數中,用print
顯示出的變量的值會是函數中的局部變量的值。如果此時你想查看全局變量的值時,你可?
允褂謾?:”操作 符:

file::variable
function::variable

可以通過這種形式指定你所想查看的變量,是哪個文件中的或是哪個函數中的。例如,查?
次募2.c中的全局變量x的值:

(gdb) p 'f2.c'::x

當然,“::”操作符會和C++中的發生沖突,GDB能自動識別“::” 是否C
++的操作符,所以你不必擔心在調試C++程序時會出現異 常。

另外,需要注意的是,如果你的程序編譯時開啟了優化選項,那么在用GDB
調試被優化過的程序時,可能會發生某些變量不能訪問,或是取值錯誤 碼的情況。這個是?
苷5模蛭嘔絳蚧嶸靖哪愕某絳潁砟慍絳虻撓錁淥承潁蕹恍┪摶庖宓謀淞?
等,所以在GDB
調試這 種程序時,運行時的指令和你所編寫指令就有不一樣,也就會出現你所想象不到的?
峁6愿墩庵智榭鍪保枰詒嘁氤絳蚴憊乇氈嘁胗嘔R話憷此擔 負跛械謀嘁肫鞫?
支持編譯優化的開關,例如,GNU 的C/C++編譯器GCC,你可以使用“-gstabs
”選項來解決這個問題。關 于編譯器的參數,還請查看編譯器的使用說明文檔。

11.3 數組
有時候,你需要查看一段連續的內存空間的值。比如數組的一段,或是動態分配的數據的?
笮 D憧梢允褂肎DB
的 “@”操作符,“@”的左邊是第一個內存的地址的值,“@”的右邊則你你想查看內存的
長度。例如,你的程序中有這樣的語句:

int *array = (int *) malloc (len * sizeof (int));

于是,在GDB調試過程中,你可以以如下命令顯示出這個動態數組的取值:

p *array@len

@的左邊是數組的首地址的值,也就是變量array
所指向的內容,右邊則是數據的長度,其保存在變量len
中,其輸出結果,大約是 下面這個樣子的:

(gdb) p *array@len
$1 = {2, 4, 6, 8, 10, 12, 14, 16, 18, 20, 22, 24, 26, 28, 30, 32, 34, 36, 38,
40}

如果是靜態數組的話,可以直接用print數組名,就可以顯示數組中所有數據的內容了。

11.4 輸出格式
一般來說,GDB會根據變量的類型輸出變量的值。但你也可以自定義GDB
的輸出的格式。例如,你想輸出一個整數 的十六進制,或是二進制來查看這個整型變量的?
械奈壞那榭觥R齙秸庋憧梢允褂肎DB的數據顯示格式:

x? 按十六進制格式顯示變量。
d? 按十進制格式顯示變量。
u? 按十六進制格式顯示無符號整型。
o? 按八進制格式顯示變量。
t? 按二進制格式顯示變量。
a? 按十六進制格式顯示變量。
c? 按字符格式顯示變量。
f? 按浮點數格式顯示變量。

(gdb) p i
$21 = 101??
(gdb) p/a i
$22 = 0x65
(gdb) p/c i
$23 = 101 'e'
(gdb) p/f i
$24 = 1.41531145e-43
(gdb) p/x i
$25 = 0x65
(gdb) p/t i
$26 = 1100101

11.5查看內存
你可以使用examine命令(簡寫是x)來查看內存地址中的值。x命令的語法如下所示:

x/<n/f/u> <addr>

n、f、u是可選的參數。

n 是一個正整數,表示顯示內存的長度,也就是說從當前地址向後顯示幾個地址的內容。
f 表示顯示的格式,參見上面。如果地址所指的是字符串,那么格式可以是s
,如果地十是指令地址,那么格式可以是i。
u 表示從當前地址往後請求的字節數,如果不指定的話,GDB默認是4個bytes。u
參數可以用下面的字符來代替,b表示單字節,h表示雙字節,w 表示四字節,g
表示八字節。當我們指定了字節長度後,GDB
會從指內存定的內存地址開始,讀寫指定字節,并把其當作一個值取出來。
<addr> 表示一個內存地址。

n/f/u三個參數可以一起使用。例如:

命令:x/3uh 0x54320 表示,從內存地址0x54320讀取內容,h表示以雙字節為一個單位,
3表示三個單位,u表示按十六進 制顯示。

11.6 自動顯示
你可以設置一些自動顯示的變量,當程序停住時,或是在你單步跟蹤時,這些變量會自動?
允盡O喙氐腉DB命令是 display。

display <expr>
display/<fmt> <expr>
display/<fmt> <addr>

expr是一個表達式,fmt表示顯示的格式,addr表示內存地址,當你用display
設定好了一個或多個表達式後,只要你的程序被停 下來,GDB
會自動顯示你所設置的這些表達式的值。

格式i和s同樣被display支持,一個非常有用的命令是:

display/i $pc

$pc是GDB的環境變量,表示著指令的地址,/i
則表示輸出格式為機器指令碼,也就是匯編。于是當程序停下後,就會出現源代碼和機器?
噶 盥胂嘍雜Φ那樾危饈且桓齪苡幸饉嫉墓δ堋?

下面是一些和display相關的GDB命令:

undisplay <dnums...>
delete display <dnums...>

刪除自動顯示,dnums
意為所設置好了的自動顯式的編號。如果要同時刪除幾個,編號可以用空格分隔,如果要?
境桓齜段詰謀嗪 牛梢雜眉鹺瘧硎荊ㄈ紓?-5)

disable display <dnums...>
enable display <dnums...>

disable和enalbe不刪除自動顯示的設置,而只是讓其失效和恢復。

info display

查看display設置的自動顯示的信息。GDB
會打出一張表格,向你報告當然調試中設置了多少個自動顯示設置,其中包括,設置的編?
牛 澩鍤劍欠馿nable。

11.7 設置顯示選項
GDB中關于顯示的選項比較多,這里我只例舉大多數常用的選項。

set print address
set print address on
打開地址輸出,當程序顯示函數信息時,GDB
會 顯出函數的參數地址。系統默認為打開的,如:
(gdb) f
#0? set_quotes (lq=0x34c78 "<<", rq=0x34c88 ">>")
??? at input.c:530
530???????? if (lquote != def_lquote)

set print address off
關閉函數的參數地址顯示,如:
(gdb) set print addr off
(gdb) f
#0? set_quotes (lq="<<", rq=">>") at input.c:530
530???????? if (lquote != def_lquote)

show print address
查看當前地址顯示選項是否打開。
set print array
set print array on
打開數組顯示,打開後當數組顯示時,每個元素占一行,如果不打開的話,每個元素則以?
漢歐指簟U飧鲅∠钅 鮮槍乇盞摹S脛喙氐牧礁雒釗縵攏揖筒輝俁嗨盜恕?
set print array off
show print array
set print elements <number-of-elements>
這個選項主要是設置數組的,如果你的數組太大了,那么 就可以指定一個<number-of-
elements>來指定數據顯示的最大長度,當到達這個長度時,GDB
就不再往下顯 示了。如果設置為0,則表示不限制。
show print elements
查看print elements的選項信息。
set print null-stop <on/off>
如果打開了這個選項,那么當顯示字符串時,遇到結束符則停止顯示。這個選項默 認為
off。
set print pretty on
如果打開printf pretty這個選項,那么當GDB顯示結構體時會比較漂亮。如:
$1 = {
? next = 0x0,
? flags = {
??? sweet = 1,
??? sour = 1
? },
? meat = 0x54 "Pork"
}

set print pretty off
關閉printf pretty這個選項,GDB顯示結構體時會如下顯示:
$1 = {next = 0x0, flags = {sweet = 1, sour = 1}, meat = 0x54 "Pork"}

show print pretty
查看GDB是如何顯示結構體的。
set print sevenbit-strings <on/off>
設置字符顯示,是否按“/nnn”的格式顯示,如果打開,則字符串或字符數據按/nnn
顯示,如 “/065”。
show print sevenbit-strings
查看字符顯示開關是否打開。
set print union <on/off>
設置顯示結構體時,是否顯式其內的聯合體數據。例如有以下數據結構:
typedef enum {Tree, Bug} Species;
typedef enum {Big_tree, Acorn, Seedling} Tree_forms;
typedef enum {Caterpillar, Cocoon, Butterfly}
????????????? Bug_forms;

struct thing {
? Species it;
? union {
??? Tree_forms tree;
??? Bug_forms bug;
? } form;
};

struct thing foo = {Tree, {Acorn}};

當打開這個開關時,執行 p foo 命令後,會如下顯示:

$1 = {it = Tree, form = {tree = Acorn, bug = Cocoon}}

當關閉這個開關時,執行 p foo 命令後,會如下顯示:

$1 = {it = Tree, form = {...}}

show print union
查看聯合體數據的顯示方式
set print object <on/off>:在C
++中,如果一個對象指針指向其派生類,如果打開這個選項,GDB
會自動按照虛方法調用的規則顯示輸 出,如果關閉這個選項的話,GDB
就不管虛函數表了。這個選項默認是off。
show print object
查看對象選 項的設置。
set print static-members <on/off>
這個選項表示,當顯示一個C++對象中 的內容是,是否顯示其中的靜態數據成員。默認是
on。
show print static-members
查看靜態數據成員 選項設置。
set print vtbl <on/off>
當此選項打開時,GDB將用比較規整的格式來顯示虛函數表 時。其默認是關閉的。
show print vtbl
查看虛函數顯示格式的選項。
11.8 歷史記錄
當你用GDB 的print查看程序運行時的數據時,你每一個print都會被GDB記錄下來。GDB
會以$1, $2, $3 .....這樣的方式為你每一個print
命令編上號。于是,你可以使用這個編號訪問以前的表達式,如$1
。這個功能所帶來的好處是,如 果你先前輸入了一個比較長的表達式,如果你還想查看這?
霰澩鍤降鬧擔憧梢允褂美芳鍬祭捶夢剩∪チ酥馗詞淙搿?

11.9 GDB環境變量
你可以在GDB
的調試環境中定義自己的變量,用來保存一些調試程序中的運行數據。要定義一個GDB
的 變量很簡單只需。使用GDB的set命令。GDB的環境變量和UNIX一樣,也是以$起頭。如:

set $foo = *object_ptr

使用環境變量時,GDB
會在你第一次使用時創建這個變量,而在以後的使用中,則直接對其賦值。環境變量沒有?
嘈停憧梢愿肪潮淞 慷ㄒ迦我壞睦嘈汀0ń峁固搴褪欏?

show convenience
該命令查看當前所設置的所有的環境變量。
這是一個比較強大的功能,環境變量和程序變量的交互 使用,將使得程序調試更為靈活便?
蕁@紓?

set $i = 0
print bar[$i++]->contents

于是,當你就不必,print bar[0]->contents, print bar[1]->contents
地輸入命 令了。輸入這樣的命令後,只用敲回車,重復執行上一條語句,環境變量會自動?
奐櫻傭瓿芍鷥鍪涑齙墓δ堋?

11.10 查看寄存器
要查看寄存器的值,很簡單,可以使用如下命令:

info registers
查看寄存器的情況。(除了浮點寄存器)
info all-registers
查看所有 寄存器的情況。(包括浮點寄存器)
info registers <regname ...>
查看所指定的寄存器的情 況。
寄存器中放置了程序運行時的數據,比如程序當前運行的指令地址(ip
),程序的當前堆棧地址(sp)等等。你同樣可以使用 print
命令來訪問寄存器的情況,只需要在寄存器名字前加一個$符號就可以了。如:p $eip。

12. 改變程序的執行
一旦使用GDB
掛上被調試程序,當程序運行起來後,你可以根據自己的調試思路來動態地在GDB
中 更改當前被調試程序的運行線路或是其變量的值,這個強大的功能能夠讓你更好的調試?
愕某絳潁熱紓憧梢栽誄絳虻囊淮臥誦兄兇弒槌絳虻乃蟹種 А?

12.1 修改變量值
修改被調試程序運行時的變量值,在GDB中很容易實現,使用GDB的print
命令即可完成。如:

(gdb) print x=4

x=4這個表達式是C/C++的語法,意為把變量x的值修改為4,如果你當前調試的語言是
Pascal,那么你可以使用Pascal的語 法:x:=4。

在某些時候,很有可能你的變量和GDB中的參數沖突,如:

(gdb) whatis width
type = double
(gdb) p width
$4 = 13
(gdb) set width=47
Invalid syntax in expression.

因為,set width是GDB的命令,所以,出現了“Invalid syntax in expression
”的設置錯誤,此時, 你可以使用set var命令來告訴GDB,width不是你GDB
的參數,而是程序的變量名,如:

(gdb) set var width=47

另外,還可能有些情況,GDB
并不報告這種錯誤,所以保險起見,在你改變程序變量取值時,最好都使用set var
格式的GDB命 令。

12.2跳轉執行
一般來說,被調試程序會按照程序代碼的運行順序依次執行。GDB
提供了亂序執行的功能,也就是說,GDB
可 以修改程序的執行順序,可以讓程序執行隨意跳躍。這個功能可以由GDB的jump
命令來完:

jump <linespec>
指定下一條語句的運行點。<linespce>可以是文件的行號,可以是 file:line
格式,可以是+num這種偏移量格式。表式著下一條運行語句從哪里開始。
jump <address>
這里的<address>是代碼行的內存地址。
注意,jump
命令不會改變當前的程序棧中的內容,所以,當你從 一個函數跳到另一個函數時,當函數?
誦型攴禱厥苯械徊僮魘北厝換岱⑸砦螅贍芙峁故欠淺F婀值模踔劣誆絳?
Core Dump。所以最好是同一個函數中進行跳轉。

熟悉匯編的人都知道,程序運行時,有一個寄存器用于保存當前代碼所在的內存地址。所?
裕琷ump命令也就是改變了這個寄存器中的值。于是, 你可以使用“set $pc
”來更改跳轉執行的地址。如:

set $pc = 0x485

12.3產生信號量
使用singal命令,可以產生一個信號量給被調試的程序。如:中斷信號Ctrl+C
。這非常方便于程序的調 試,可以在程序運行的任意位置設置斷點,并在該斷點用GDB
產生一個信號量,這種精確地在某處產生信號非常有利程序的調試。

語法是:signal <singal>,UNIX的系統信號量通常從1到15。所以<singal
>取值也在這 個范圍。

single命令和shell的kill命令不同,系統的kill命令發信號給被調試程序時,是由GDB
截獲的,而single命令所發出 一信號則是直接發給被調試程序的。

12.4強制函數返回
如果你的調試斷點在某個函數中,并還有語句沒有執行完。你可以使用return
命令強制函數忽略還沒有執行 的語句并返回。

return
return <expression>
?? 使用return命令取消當前函數的執行,并立即返回,如果指定了<expression
>,那么該表達式的值會被認作函數的返回 值。

12.5 強制調用函數
call <expr>

表達式中可以一是函數,以此達到強制調用函數的目的。并顯示函數的返回值,如果函數?
禱刂凳莢oid,那么就不顯示。

另一個相似的命令也可以完成這一功能——print,print
後面可以跟表達式,所以也可以用他來調用函數,print和call的不同 是,如果函數返回
void,call則不顯示,print則顯示函數返回值,并把該值存入歷史數據中。

13. 在不同語言中使用GDB
GDB支持下列語言:C, C++, Fortran, PASCAL, Java, Chill, assembly, 和 Modula-2
。一般說來,GDB
會根據你所調試的程序來確定當然的調試語言,比如:發現文件名後綴為 “.c”的,GDB
會認為是C程序。文件名後綴為 “.C, .cc, .cp, .cpp, .cxx, .c++”的,GDB會認為是C
++ 程序。而後綴是“.f, .F”的,GDB會認為是Fortran程序,還有,後綴為如果是“.s,
.S”的會認為是匯編語言。

也就是說,GDB會根據你所調試的程序的語言,來設置自己的語言環境,并讓GDB
的命令跟著語言環境的改變而改變。比如一些GDB
命 令需要用到表達式或變量時,這些表達式或變量的語法,完全是根據當前的語言環境而?
謀淶摹@鏑/C++中對指針的語法是*p,而在 Modula-2中則是p
^。并且,如果你當前的程序是由幾種不同語言一同編譯成的,那到在調試過程中,GDB
也能根據不同的語言自動地 切換語言環境。這種跟著語言環境而改變的功能,真是體貼開?
⑷嗽鋇囊恢稚杓啤?

下面是幾個相關于GDB語言環境的命令:

show language
查看當前的語言環境。如果GDB不能識為你所調試的編程語言,那么,C
語言被認為是默認的環境。
info frame
查看當前函數的程序語言。
info source
查看當前文件的程序語言。
如果GDB沒有檢測出當 前的程序語言,那么你也可以手動設置當前的程序語言。使用set
language命令即可做到。

當set language命令後什么也不跟的話,你可以查看GDB所支持的語言種類:

(gdb) set language
The currently understood settings are:

local or auto??? Automatic setting based on source file
c??????????????? Use the C language
c++????????????? Use the C++ language
asm????????????? Use the Asm language
chill??????????? Use the Chill language
fortran????????? Use the Fortran language
java???????????? Use the Java language
modula-2???????? Use the Modula-2 language
pascal?????????? Use the Pascal language
scheme?????????? Use the Scheme language

于是你可以在set language後跟上被列出來的程序語言名,來設置當前的語言環境。

14. 後記
GDB
是一個強大的命令行調試工具。大家知道命令行的強大就是在于,其可以形成執行序列,?
緯山瘧盡 NIX
下的軟件全是命令行的,這給程序開發提代供了極大的便利,命令行軟件的優勢在于,它?
強梢苑淺H菀椎募稍諞黃穡褂眉父黽虻サ 囊延泄ぞ叩拿睿涂梢宰齔鲆桓齜淺G看?
的功能。

于是UNIX下的軟件比Windows
下的軟件更能有機地結合,各自發揮各自的長處,組合成更為強勁的功能。而Windows
下的 圖形軟件基本上是各自為營,互相不能調用,很不利于各種軟件的相互集成。在這里?
⒉皇且蚖indows
做個什么比較,所謂“寸有所長, 尺有所短”,圖形化工具還是有不如命令行的地方。(?
吹秸餼浠笆保M魑磺蛟僖膊灰銜揖褪恰氨墑油夾謂緱妗保臀姨Ц芰?)

我是根據版本為5.1.1的GDB
所寫的這篇文章,所以可能有些功能已被修改,或是又有更為強勁的功能。而且,我寫得?
淺2執伲 吹帽冉霞蚵裕⑶遙渲形乙丫吹接行磯啻肀鹱至耍ㄎ矣夢灞剩源磣秩?
你看不懂),所以,我在這里對我文中的差錯表示萬分的歉意。

文中所羅列的GDB的功能時,我只是羅列了一些帶用的GDB
的命令和使用方法,其實,我這里只講述的功能大約只占GDB所有功能的60
% 吧,詳細的文檔,還是請查看GDB
的幫助和使用手冊吧,或許,過段時間,如果我有空,我再寫一篇GDB的高級使用。

我個人非常喜歡GDB的自動調試的功能,這個功能真的很強大,試想,我在UNIX
下寫個腳本,讓腳本自動編譯我的程序,被自動調試,并把結 果報告出來,調試成功,自?
痗heckin源碼庫。一個命令,編譯帶著調試帶著checkin,多爽啊。只是GDB
對自動化調試目前支持 還不是很成熟,只能實現半自動化,真心期望著GDB
的自動化調試功能的成熟。

如果各位對GDB或是別的技術問題有興趣的話,歡迎和我討論交流。本人目前主要在UNIX
下做產品軟件的開發,所以,對UNIX
下 的軟件開發比較熟悉,當然,不單單是技術,對軟件工程實施,軟件設計,系統分析,?
钅抗芾砦乙猜雜行牡謾;隊蠹藝椅醫渙鰨≦Q 是:753640,MSN 是:haoel@hotmail .
com)

[編輯] 相關詞條
GCC新手入門
C/C++ IDE簡介
用GDB調試程序
Gtk與Qt編譯環境安裝與配置
跟我一起寫Makefile
?C編譯初步
C++編譯初步
Fortran編譯初步
C和C++混合編譯初 步
C和Fortran混合編譯初步

取自"http://wiki.ubuntu.org.cn/index.php?title=%E7%94%A8GDB%E8%B0%83%E8%AF%95%
E7%A8%8B%E5%BA%8F&variant=zh-cn"
1 個分類: 程序開發


本文轉自
http://wiki.ubuntu.org.cn/index.php?title=%E7%94%A8GDB%E8%B0%83%E8%AF%95%E7%A8
%8B%E5%BA%8F&variant=zh-cn

總結

以上是生活随笔為你收集整理的GDB调试程序-Ubuntu的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。