GDB是GNU開源組織釋出的一個強大的UNIX下的程式調試工具。如果你是在 UNIX平台下做軟體,你會發現GDB這個調試工具有比VC、BCB的圖形化調試器更強大的功能。同時GDB也具有例如ddd這樣的圖形化的調試端。
一般來說,GDB主要完成下面四個方面的功能:
(1)啟動你的程式,可以按照你的自定義的要求随心所欲的運作程式。
(2)可讓被調試的程式在你所指定的調置的斷點處停住。(斷點可以是條件表達式)
(3)當程式被停住時,可以檢查此時你的程式中所發生的事。
(4)動态的改變你程式的執行環境。
興趣是最好的老師,這裡先整理總結一下在調試的過程中經常遇到的問題。帶着這些問題進行學習和實踐可以有助于加深印象。 再往後是本人實踐過程中總結的常見指令,如有什麼問題或者建議,都可以聯系我,謝謝!^_^
(1)如何列印變量的值?(print var)
(2)如何列印變量的位址?(print &var)
(3)如何列印位址的資料值?(print *address)
(4)如何檢視目前運作的檔案和行?(backtrace)
(5)如何檢視指定檔案的代碼?(list file:N)
(6)如何立即執行完目前的函數,但是并不是執行完整個應用程式?(finish)
(7)如果程式是多檔案的,怎樣定位到指定檔案的指定行或者函數?(list file:N)
(8)如果循環次數很多,如何執行完目前的循環?(until)
(9)多線程如何調試?(???)
作者:QuietHeart
Email: [email protected]
[舉例]
=============
*啟動gdb
$gdb
這樣可以和gdb進行互動了。
*啟動gdb,并且分屏顯示源代碼:
$gdb -tui
這樣,使用了'-tui'選項,啟動可以直接将螢幕分成兩個部分,上面顯示源代碼,比用list友善多了。這時候使用上下方向鍵可以檢視源代碼,想要指令行使用上下鍵就用[Ctrl]n和[Ctrl]p.
*啟動gdb調試指定程式app:
$gdb app
這樣就在啟動gdb之後直接載入了app可執行程式,需要注意的是,載入的app程式必須在編譯的時候有gdb調試選項,例如'gcc -g app app.c',注意,如果修改了程式的源代碼,但是沒有編譯,那麼在gdb中顯示的會是改動後的源代碼,但是運作的是改動前的程式,這樣會導緻跟蹤錯亂的。
*啟動程式之後,再用gdb調試:
$gdb <program> <PID>
這裡,<program>是程式的可執行檔案名,<PID>是要調試程式的PID.如果你的程式是一個服務程式,那麼你可以指定這個服務程式運作時的程序ID。gdb會自動attach上去,并調試他。program應該在PATH環境變量中搜尋得到。
*啟動程式之後,再啟動gdb調試:
$gdb <PID>
這裡,程式是一個服務程式,那麼你可以指定這個服務程式運作時的程序ID,<PID>是要調試程式的PID.這樣gdb就附加到程式上了,但是現在還沒法檢視源代碼,用file指令指明可執行檔案就可以顯示源代碼了。
**啟動gdb之後的互動指令:
互動指令支援[Tab]補全。
*顯示幫助資訊:
(gdb) help
*載入指定的程式:
(gdb) file app
這樣在gdb中載入想要調試的可執行程式app。如果剛開始運作gdb而不是用gdb app啟動的話可以這樣載入app程式,當然編譯app的時候要加入-g調試選項。
*重新運作調試的程式:
(gdb) run
要想運作準備調試的程式,可使用run指令,在它後面可以跟随發給該程式的任何參數,包括标準輸入和标準輸出說明符(<和> )和shell通配符(*、?、[、])在内。
*修改發送給程式的參數:
(gdb) set args no
這裡,假設我使用"r yes"設定程式啟動參數為yes,那麼這裡的set args會設定參數argv[1]為no。
*顯示預設的參數清單:
(gdb) show args
*列出指定區域(n1到n2之間)的代碼:
(gdb) list n1 n2
這樣,list可以簡寫為l,将會顯示n1行和n2行之間的代碼,如果使用-tui啟動gdb,将會在相應的位置顯示。如果沒有n1和n2參數,那麼就會預設顯示目前行和之後的10行,再執行又下滾10行。另外,list還可以接函數名。
一般來說在list後面可以跟以下這們的參數:
<linenum> 行号。
<+offset> 目前行号的正偏移量。
<-offset> 目前行号的負偏移量。
<filename:linenum> 哪個檔案的哪一行。
<function> 函數名。
<filename:function> 哪個檔案中的哪個函數。
<*address> 程式運作時的語句在記憶體中的位址。
*執行下一步:
(gdb) next
這樣,執行一行代碼,如果是函數也會跳過函數。這個指令可以簡化為n.
*執行N次下一步:
(gdb) next N
*執行上次執行的指令:
(gdb) [Enter]
這裡,直接輸入回車就會執行上次的指令了。
*單步進入:
(gdb) step
這樣,也會執行一行代碼,不過如果遇到函數的話就會進入函數的内部,再一行一行的執行。
*執行完目前函數傳回到調用它的函數:
(gdb) finish
這裡,運作程式,直到目前函數運作完畢傳回再停止。例如進入的單步執行如果已經進入了某函數,而想退出該函數傳回到它的調用函數中,可使用指令finish.
*指定程式直到退出目前循環體:
(gdb) until
或(gdb) u
這裡,發現需要把光标停止在循環的頭部,然後輸入u這樣就自動執行全部的循環了。
*跳轉執行程式到第5行:
(gdb) jump 5
這裡,可以簡寫為"j 5"需要注意的是,跳轉到第5行執行完畢之後,如果後面沒有斷點則繼續執行,而并不是停在那裡了。
另外,跳轉不會改變目前的堆棧内容,是以跳到别的函數中就會有奇怪的現象,是以最好跳轉在一個函數内部進行,跳轉的參數也可以是程式代碼行的位址,函數名等等類似list。
*強制傳回目前函數:
(gdb) return
這樣,将會忽略目前函數還沒有執行完畢的語句,強制傳回。return後面可以接一個表達式,表達式的傳回值就是函數的傳回值。
*強制調用函數:
(gdb) call <expr>
這裡,<expr>可以是一個函數,這樣就會傳回函數的傳回值,如果函數的傳回類型是void那麼就不會列印函數的傳回值,但是實踐發現,函數運作過程中的列印語句還是沒有被列印出來。
*強制調用函數2:
(gdb) print <expr>
這裡,print和call的功能類似,不同的是,如果函數的傳回值是void那麼call不會列印傳回值,但是print還是會列印出函數的傳回值并且存放到曆史記錄中。
*在目前的檔案中某一行(假設為6)設定斷點:
(gdb) break 6
*設定條件斷點:
(gdb) break 46 if testsize==100
這裡,如果testsize==100就在46行處斷點。
*檢測表達式變化則停住:
(gdb) watch i != 10
這裡,i != 10這個表達式一旦變化,則停住。watch <expr> 為表達式(變量)expr設定一個觀察點。一量表達式值有變化時,馬上停住程式(也是一種斷點)。
*在目前的檔案中為某一函數(假設為func)處設定斷點:
(gdb) break func
*給指定檔案(fileName)的某個行(N)處設定斷點:
(gdb) break fileName:N
這裡,給某檔案中的函數設定斷點是同理的。
*顯示目前gdb斷點資訊:
(gdb) info breakpoints
這裡,可以簡寫為info break.會顯示目前所有的斷點,斷點号,斷點位置等等。
*删除N号斷點:
(gdb) delete N
*删除所有斷點:
(gdb) delete
*清除行N上面的所有斷點:
(gdb) clear N
*繼續運作程式直接運作到下一個斷點:
(gdb) continue
這裡,如果沒有斷點就一直運作。
*顯示目前調用函數堆棧中的函數:
(gdb) backtrace
指令産生一張清單,包含着從最近的過程開始的所有有效過程和調用這些過程的參數。當然,這裡也會顯示出目前運作到了哪裡(檔案,行)。
*檢視目前調試程式的語言環境:
(gdb) show language
這裡,如果gdb不能識别你所調試的程式,那麼預設是c語言。
*檢視目前函數的程式語言:
(gdb) info frame
*顯示目前的調試源檔案:
(gdb) info source
這樣會顯示目前所在的源代碼檔案資訊,例如檔案名稱,程式語言等。
*手動設定目前的程式語言為c++:
(gdb) set language c++
這裡,如果gdb沒有檢測出你的程式語言,你可以這樣設定。
*檢視可以設定的程式語言:
(gdb) set language
這裡,使用沒有參數的set language可以檢視gdb中可以設定的程式語言。
*終止一個正在調試的程式:
(gdb) kill
這裡,輸入kill就會終止正在調試的程式了。
*print顯示變量(var)值:
(gdb) print var
這裡,print可以簡寫為p,print 是gdb的一個功能很強的指令,利用它可以顯示被調試的語言中任何有效的表達式。表達式除了包含你程式中的變量外,還可以包含函數調用,複雜資料結構和曆史等等。
*用16進制顯示(var)值:
(gdb) print /x var
這裡可以知道,print可以指定顯示的格式,這裡用'/x'表示16進制的格式。
可以支援的變量顯示格式有:
x 按十六進制格式顯示變量。
d 按十進制格式顯示變量。
u 按十六進制格式顯示無符号整型。
o 按八進制格式顯示變量。
t 按二進制格式顯示變量。
a 按十六進制格式顯示變量。
c 按字元格式顯示變量。
f 按浮點數格式顯示變量。
*如果a是一個數組,10個元素,如果要顯示則:
(gdb) print *a@10
這樣,會顯示10個元素,無論a是double或者是int的都會正确地顯示10個元素。
*修改運作時候的變量值:
(gdb) print x=4
這裡,x=4是C/C++的文法,意為把變量x值改為4,如果你目前調試的語言是Pascal,那麼你可以使用Pascal的文法:x:=4。
*顯示一個變量var的類型:
(gdb) whatis var
*以更詳細的方式顯示變量var的類型:
(gdb) ptype var
這裡,會列印出var的結構定義。
[其他]
*在Qt4.x環境中列印QString msg;的msg變量:
步驟如下:
1)定義一個宏printqstring
define printqstring
printf "(QString)0x%x (length=%i): \"",&$arg0,$arg0.d->size
set $i=0
while $i < $arg0.d->size
set $c=$arg0.d->data[$i++]
if $c < 32 || $c > 127
printf "\\u0x%04x", $c
else
printf "%c", (char)$c
end
end
printf "\"\n"
end
2)(gdb) printqstring msg
這裡,這個宏可以在gdb中直接定義,據說也可以寫到$HOME/.gdbinit,這樣每次啟動自動加載。
*調試同時指明生成core檔案:
$gdb <program> core
用gdb同時調試一個運作程式和core檔案,core是程式非法執行後core dump後産生的檔案。當程式非法崩潰的時候會産生一個core檔案,然後使用這個指令,會直接定位到發生程式崩潰的位置。注意:有時需要設定系統指令“ulimit -c unlimited”才能産生core檔案。
**沒有實踐過的
*print顯示存儲塊,如顯示h後面的10個整數:
print h@10
**
本文轉自莫水千流部落格園部落格,原文連結:http://www.cnblogs.com/zhoug2020/p/7283169.html,如需轉載請自行聯系原作者