看執行時資料
———————
在你除錯程式時,當程式被停住時,你可以使用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除錯程式(五)
檢視執行時資料 在你除錯程式時,當程式被停住時,你可以使用print命令 簡寫命令為p 或是同義命令inspect來檢視當前程式的執行資料。print命令的格式是 print print 是表示式,是你所除錯的程式的語言的表示式 gdb可以除錯多種程式語言 是輸出的格式,比如,如果要把表示式按16進...
用GDB除錯程式(五)
檢視執行時資料 在你除錯程式時,當程式被停住時,你可以使用print命令 簡寫命令為p 或是同義命令inspect來檢視當前程式的執行資料。print命令的格式是 print print 是表示式,是你所除錯的程式的語言的表示式 gdb可以除錯多種程式語言 是輸出的格式,比如,如果要把表示式按16進...
用GDB除錯程式(五)
檢視執行時資料 在你除錯程式時,當程式被停住時,你可以使用print命令 簡寫命令為p 或是同義命令inspect來檢視當前程式的執行資料。print命令的格式是 print print 是表示式,是你所除錯的程式的語言的表示式 gdb可以除錯多種程式語言 是輸出的格式,比如,如果要把表示式按16進...