用GDB除錯程式 檢視執行時資料

2021-05-22 22:54:16 字數 4300 閱讀 2101

參考自:http://hi.baidu.com/wg_wang/blog/item/dad263c2f5598630e5dd3ba6.html

在使用gdb除錯程式時,觸發斷點後,可以使用

print

命令(簡寫為

p),或是同義命令

inspect

來檢視當前程式的執行資料。

print

命令的格式是:    

print

print /

是表示式,是所除錯程式的語言的表示式(

gdb可以除錯多種程式語言);

是輸出的格式,例如如果要把表示式按

16進製制的格式輸出,那麼就是/x。

print

和許多gdb

的命令一樣,可以接受乙個表示式,

gdb會根據當前的程式執行的資料來計算這個表示式,既然是表示式,那麼就可以是當前程式執行中的

const

常量、變數、函式等內容。可惜的是

gdb不能使用你在程式中所定義的巨集(因為巨集是在編譯時就已經展開了的)。表示式的語法應該是當前所除錯的語言的語法。

在表示式中,有幾種

gdb所支援的操作符,它們可以用在任何一種語言中。    

@        

是乙個和陣列有關的操作符,在後面會有更詳細的說明。        

::         

指定乙個在檔案或是乙個函式中的變數。        

{} 表示乙個指向記憶體位址

的型別為

type

的乙個物件。       

在gdb

中,你可以隨時檢視以下三種變數的值:

1、全域性變數(所有檔案可見的)

2、靜態全域性變數(當前檔案可見的)

3、區域性變數(當前

scope

可見的)        

如果區域性變數和全域性變數發生衝突(也就是重名),一般情況下是區域性變數會隱藏全域性變數,也就是說,用

print

顯示出的變數的值會是函式中的區域性變數的值。如果此時想檢視全域性變數的值,可以使用「

::」操作符:    

file::variable

function::variable

可以通過這種形式指定想檢視的變數是哪個檔案中的或是哪個函式中的。例如,檢視檔案

f2.c

中的全域性變數

x的值:

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

當然,「

::」操作符會和

c++中的發生衝突,

gdb能自動識別「

::」 

是否c++

的操作符,所以不必擔心在除錯

c++程式時會出現異常。    

另外,需要注意的是,如果程式編譯時開啟了優化選項,那麼在用

gdb除錯被優化過的程式時,可能會發生某些變數不能訪問,或是取值錯誤碼的情況。這是因為優化程式會刪改你的程式,整理你程式的語句順序,剔除一些無意義的變數等,所以在

gdb除錯這種程式時,執行時的指令和所編寫指令就有不一樣,也就會出現所想象不到的結果。對付這種情況時,可以在編譯程式時關閉編譯優化。一般來說,幾乎所有的編譯器都支援編譯優化的開關,例如,

gnu的

c/c++

編譯器gcc

,你可以使用「

-gstabs」

選項來解決這個問題。關   

有時候,需要檢視一段連續的記憶體空間的值。比如陣列的一段,或是動態分配的資料的大小。可以使用

gdb的「

@」操作符,「

@」的左邊是第乙個記憶體的位址的值,「

@」的右邊是想檢視記憶體的長度。例如,程式中有這樣的語句:     

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

於是,在

gdb除錯過程中,你可以以如下命令顯示出這個動態陣列的取值:

p *array@len

@的左邊是陣列的首位址的值,也就是變數

array

所指向的內容,右邊則是資料的長度,其儲存在變數

len中,其輸出結果,大約是下面這個樣子的:    

(gdb) p *array@len

$1 =

如果是靜態陣列的話,可以直接用

print

陣列名,就可以顯示陣列中所有資料的內容了。

一般來說,

gdb會根據變數的型別輸出變數的值。但也可以自定義

gdb的輸出的格式。例如,想輸出乙個整數的十六進製制,或是二進位制來檢視這個整型變數的中的位的情況。可以使用

gdb的資料顯示格式:    

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

可以使用

examine

命令(簡寫是

x)來檢視記憶體位址中的值。

x命令的語法如下所示:    x/n

、f、u

是可選的引數。    

n 是乙個正整數,表示顯示記憶體的長度,也就是說從當前位址向後顯示幾個位址的內容。

f 表示顯示的格式,參見上面。如果位址所指的是字串,那麼格式可以是

s,如果地十是指令位址,那麼格式可以是i。

u 表示從當前位址往後請求的位元組數,如果不指定的話,

gdb預設是4個

bytes。u

引數可以用下面的字元來代替,

b表示單位元組,

h表示雙位元組,

w表示四位元組,

g表示八字節。當我們指定了位元組長度後,

gdb會從指記憶體定的記憶體位址開始,讀寫指定位元組,並把其當作乙個值取出來。    

表示乙個記憶體位址。

n/f/u

三個引數可以一起使用。例如:    

命令:x/3uh 0x54320 

表示,從記憶體位址

0x54320

讀取內容,

h表示以雙位元組為乙個單位,

3表示三個單位,

u表示按十六進製制顯示。   

可以設定一些自動顯示的變數,當程式停住時,或是在單步跟蹤時,這些變數會自動顯示。相關的

gdb命令是

display

。display

display/

display/

expr

是乙個表示式,

fmt表示顯示的格式,

addr

表示記憶體位址,當用

display

設定好了乙個或多個表示式後,只要你的程式被停下來,

gdb會自動顯示你所設定的這些表示式的值。    格式i

和s同樣被display

支援,乙個非常有用的命令是:    

display/i $pc    

$pc是

gdb的環境變數,表示著指令的位址,

/i則表示輸出格式為機器指令碼,也就是彙編。於是當程式停下後,就會出現源**和機器指令碼相對應的情形,這是乙個很有意思的功能。    

下面是一些和

display

相關的gdb

命令:    

undisplay

delete display

刪除自動顯示,

dnums

意為所設定好了的自動顯式的編號。如果要同時刪除幾個,編號可以用空格分隔,如果要刪除乙個範圍內的編號,可以用減號表示(如:

2-5)    

disable display

enable display

disable

和enalbe

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

info display

檢視display

設定的自動顯示的資訊。

gdb會打出一張**,向你報告當然除錯中設定了多少個自動顯示設定,其中包括,設定的編號,表示式,是否

enable。

用GDB除錯程式 10 檢視執行時資料(1)

用gdb除錯程式 10 檢視執行時資料 1 在你除錯程式時,當程式被停住時,你可以使用print命令 簡寫命令為p 或是同義命令inspect來檢視當前程式的執行資料。print命令的格式是 print print 是表示式,是你所除錯的程式的語言的表示式 gdb可以除錯多種程式語言 是輸出的格式,...

用GDB除錯程式

用gdb除錯程式 gdb概述 gdb是gnu開源組織發布的乙個強大的unix下的程式除錯工具。或許,各位比較喜歡那種圖形介面方式的,像vc bcb等ide的除錯,但如果你是在unix平台下做軟體,你會發現gdb這個除錯工具有比vc bcb的圖形化偵錯程式更強大的功能。所謂 寸有所長,尺有所短 就是這...

用GDB除錯程式

七 設定顯示選項 gdb中關於顯示的選項比較多,這裡我只例舉大多數常用的選項。set print address set print address on 開啟位址輸出,當程式顯示函式資訊時,gdb會顯出函式的引數位址。系統預設為開啟的,如 gdb f 0 set quotes lq 0x34c78...