查看運(yùn)行時(shí)數(shù)據(jù)
———————
在你調(diào)試程序時(shí),當(dāng)程序被停住時(shí),你可以使用print命令(簡(jiǎn)寫(xiě)命令為p),或是同義命令inspect來(lái)查看當(dāng)前程序的運(yùn)行數(shù)據(jù)。print命令的格式是:
print <expr>
print /<f> <expr>
<expr>是表達(dá)式,是你所調(diào)試的程序的語(yǔ)言的表達(dá)式(GDB可以調(diào)試多種編程語(yǔ)言),<f>是輸出的格式,比如,如果要把表達(dá)式按16進(jìn)制的格式輸出,那么就是/x。
一、表達(dá)式
print和許多GDB的命令一樣,可以接受一個(gè)表達(dá)式,GDB會(huì)根據(jù)當(dāng)前的程序運(yùn)行的數(shù)據(jù)來(lái)計(jì)算這個(gè)表達(dá)式,既然是表達(dá)式,那么就可以是當(dāng)前程序運(yùn)行中的const常量、變量、函數(shù)等內(nèi)容。可惜的是GDB不能使用你在程序中所定義的宏。
表達(dá)式的語(yǔ)法應(yīng)該是當(dāng)前所調(diào)試的語(yǔ)言的語(yǔ)法,由于C/C++是一種大眾型的語(yǔ)言,所以,本文中的例子都是關(guān)于C/C++的。(而關(guān)于用GDB調(diào)試其它語(yǔ)言的章節(jié),我將在后面介紹)
在表達(dá)式中,有幾種GDB所支持的操作符,它們可以用在任何一種語(yǔ)言中。
@
是一個(gè)和數(shù)組有關(guān)的操作符,在后面會(huì)有更詳細(xì)的說(shuō)明。
::
指定一個(gè)在文件或是一個(gè)函數(shù)中的變量。
{<type>} <addr>
表示一個(gè)指向內(nèi)存地址<addr>的類型為type的一個(gè)對(duì)象。
二、程序變量
在GDB中,你可以隨時(shí)查看以下三種變量的值:
1、全局變量(所有文件可見(jiàn)的)
2、靜態(tài)全局變量(當(dāng)前文件可見(jiàn)的)
3、局部變量(當(dāng)前Scope可見(jiàn)的)
如果你的局部變量和全局變量發(fā)生沖突(也就是重名),一般情況下是局部變量會(huì)隱藏全局變量,也就是說(shuō),如果一個(gè)全局變量和一個(gè)函數(shù)中的局部變量同名時(shí),如果當(dāng)前停止點(diǎn)在函數(shù)中,用print顯示出的變量的值會(huì)是函數(shù)中的局部變量的值。如果此時(shí)你想查看全局變量的值時(shí),你可以使用“::”操作符:
file::variable
function::variable
可以通過(guò)這種形式指定你所想查看的變量,是哪個(gè)文件中的或是哪個(gè)函數(shù)中的。例如,查看文件f2.c中的全局變量x的值:
gdb) p 'f2.c'::x
當(dāng)然,“::”操作符會(huì)和C++中的發(fā)生沖突,GDB能自動(dòng)識(shí)別“::” 是否C++的操作符,所以你不必?fù)?dān)心在調(diào)試C++程序時(shí)會(huì)出現(xiàn)異常。
另外,需要注意的是,如果你的程序編譯時(shí)開(kāi)啟了優(yōu)化選項(xiàng),那么在用GDB調(diào)試被優(yōu)化過(guò)的程序時(shí),可能會(huì)發(fā)生某些變量不能訪問(wèn),或是取值錯(cuò)誤碼的情況。這個(gè)是很正常的,因?yàn)閮?yōu)化程序會(huì)刪改你的程序,整理你程序的語(yǔ)句順序,剔除一些無(wú)意義的變量等,所以在GDB調(diào)試這種程序時(shí),運(yùn)行時(shí)的指令和你所編寫(xiě)指令就有不一樣,也就會(huì)出現(xiàn)你所想象不到的結(jié)果。對(duì)付這種情況時(shí),需要在編譯程序時(shí)關(guān)閉編譯優(yōu)化。一般來(lái)說(shuō),幾乎所有的編譯器都支持編譯優(yōu)化的開(kāi)關(guān),例如,GNU的C/C++編譯器GCC,你可以使用“-gstabs”選項(xiàng)來(lái)解決這個(gè)問(wèn)題。關(guān)于編譯器的參數(shù),還請(qǐng)查看編譯器的使用說(shuō)明文檔。
三、數(shù)組
有時(shí)候,你需要查看一段連續(xù)的內(nèi)存空間的值。比如數(shù)組的一段,或是動(dòng)態(tài)分配的數(shù)據(jù)的大小。你可以使用GDB的“@”操作符,“@”的左邊是第一個(gè)內(nèi)存的地址的值,“@”的右邊則你你想查看內(nèi)存的長(zhǎng)度。例如,你的程序中有這樣的語(yǔ)句:
int *array = (int *) malloc (len * sizeof (int));
于是,在GDB調(diào)試過(guò)程中,你可以以如下命令顯示出這個(gè)動(dòng)態(tài)數(shù)組的取值:
p *array@len
@的左邊是數(shù)組的首地址的值,也就是變量array所指向的內(nèi)容,右邊則是數(shù)據(jù)的長(zhǎng)度,其保存在變量len中,其輸出結(jié)果,大約是下面這個(gè)樣子的:
(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}
如果是靜態(tài)數(shù)組的話,可以直接用print數(shù)組名,就可以顯示數(shù)組中所有數(shù)據(jù)的內(nèi)容了。
四、輸出格式
一般來(lái)說(shuō),GDB會(huì)根據(jù)變量的類型輸出變量的值。但你也可以自定義GDB的輸出的格式。例如,你想輸出一個(gè)整數(shù)的十六進(jìn)制,或是二進(jìn)制來(lái)查看這個(gè)整型變量的中的位的情況。要做到這樣,你可以使用GDB的數(shù)據(jù)顯示格式:
x 按十六進(jìn)制格式顯示變量。
d 按十進(jìn)制格式顯示變量。
u 按十六進(jìn)制格式顯示無(wú)符號(hào)整型。
o 按八進(jìn)制格式顯示變量。
t 按二進(jìn)制格式顯示變量。
a 按十六進(jìn)制格式顯示變量。
c 按字符格式顯示變量。
f 按浮點(diǎn)數(shù)格式顯示變量。
(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
五、查看內(nèi)存
你可以使用examine命令(簡(jiǎn)寫(xiě)是x)來(lái)查看內(nèi)存地址中的值。x命令的語(yǔ)法如下所示:
x/<n/f/u> <addr>
n、f、u是可選的參數(shù)。
n 是一個(gè)正整數(shù),表示顯示內(nèi)存的長(zhǎng)度,也就是說(shuō)從當(dāng)前地址向后顯示幾個(gè)地址的內(nèi)容。
f 表示顯示的格式,參見(jiàn)上面。如果地址所指的是字符串,那么格式可以是s,如果地十是指令地址,那么格式可以是i。
u 表示從當(dāng)前地址往后請(qǐng)求的字節(jié)數(shù),如果不指定的話,GDB默認(rèn)是4個(gè)bytes。u參數(shù)可以用下面的字符來(lái)代替,b表示單字節(jié),h表示雙字節(jié),w表示四字節(jié),g表示八字節(jié)。當(dāng)我們指定了字節(jié)長(zhǎng)度后,GDB會(huì)從指內(nèi)存定的內(nèi)存地址開(kāi)始,讀寫(xiě)指定字節(jié),并把其當(dāng)作一個(gè)值取出來(lái)。
<addr>表示一個(gè)內(nèi)存地址。
n/f/u三個(gè)參數(shù)可以一起使用。例如:
命令:x/3uh 0x54320 表示,從內(nèi)存地址0x54320讀取內(nèi)容,h表示以雙字節(jié)為一個(gè)單位,3表示三個(gè)單位,u表示按十六進(jìn)制顯示。
六、自動(dòng)顯示
你可以設(shè)置一些自動(dòng)顯示的變量,當(dāng)程序停住時(shí),或是在你單步跟蹤時(shí),這些變量會(huì)自動(dòng)顯示。相關(guān)的GDB命令是display。
display <expr>
display/<fmt> <expr>
display/<fmt> <addr>
expr是一個(gè)表達(dá)式,fmt表示顯示的格式,addr表示內(nèi)存地址,當(dāng)你用display設(shè)定好了一個(gè)或多個(gè)表達(dá)式后,只要你的程序被停下來(lái),GDB會(huì)自動(dòng)顯示你所設(shè)置的這些表達(dá)式的值。
格式i和s同樣被display支持,一個(gè)非常有用的命令是:
display/i $pc
$pc是GDB的環(huán)境變量,表示著指令的地址,/i則表示輸出格式為機(jī)器指令碼,也就是匯編。于是當(dāng)程序停下后,就會(huì)出現(xiàn)源代碼和機(jī)器指令碼相對(duì)應(yīng)的情形,這是一個(gè)很有意思的功能。
下面是一些和display相關(guān)的GDB命令:
undisplay <dnums...>
delete display <dnums...>
刪除自動(dòng)顯示,dnums意為所設(shè)置好了的自動(dòng)顯式的編號(hào)。如果要同時(shí)刪除幾個(gè),編號(hào)可以用空格分隔,如果要?jiǎng)h除一個(gè)范圍內(nèi)的編號(hào),可以用減號(hào)表示(如:2-5)
disable display <dnums...>
enable display <dnums...>
disable和enalbe不刪除自動(dòng)顯示的設(shè)置,而只是讓其失效和恢復(fù)。
info display
查看display設(shè)置的自動(dòng)顯示的信息。GDB會(huì)打出一張表格,向你報(bào)告當(dāng)然調(diào)試中設(shè)置了多少個(gè)自動(dòng)顯示設(shè)置,其中包括,設(shè)置的編號(hào),表達(dá)式,是否enable。
本文轉(zhuǎn)自:http://blog.csdn.net/haoel/article/details/2883
posted on 2013-11-28 12:14
王海光 閱讀(493)
評(píng)論(0) 編輯 收藏 引用 所屬分類:
Linux