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

2022-05-23 06:33:09 字數 3727 閱讀 7158

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

在你除錯程式時,當程式被停住時,你可以使用print命令(簡寫命令為p),或是同義命令inspect來檢視當前程式的執行資料。print命令的格式是:   

print

print /

是表示式,是你所除錯的程式的語言的表示式(gdb可以除錯多種程式語言),是輸出的格式,比如,如果要把表示式按16進製制的格式輸出,那麼就是/x。

print和許多gdb的命令一樣,可以接受乙個表示式,gdb會根據當前的程式執行的資料來計算這個表示式,既然是表示式,那麼就可以是當前程式執行中的const常量、變數、函式等內容。可惜的是gdb不能使用你在程式中所定義的巨集。表示式的語法應該是當前所除錯的語言的語法,由於c/c++是一種大眾型的語言,所以,本文中的例子都是關於c/c++的。(而關於用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除錯程式 檢視執行時資料

參考自 http hi.baidu.com wg wang blog item dad263c2f5598630e5dd3ba6.html 在使用gdb除錯程式時,觸發斷點後,可以使用 print 命令 簡寫為 p 或是同義命令 inspect 來檢視當前程式的執行資料。print 命令的格式是 p...

用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...