天天看點

gdb基本指令(非常詳細)

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,如需轉載請自行聯系原作者

繼續閱讀