用gdb除錯程式(10)──檢視執行時資料(1)
在你除錯程式時,當程式被停住時,你可以使用print命令(簡寫命令為p),或是同義命令inspect來檢視當前程式的執行資料。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...