Linux下C編程基礎(chǔ)之:gdb調(diào)試器
(2)設(shè)置斷點。
設(shè)置斷點是調(diào)試程序中一個非常重要的手段,它可以使程序運(yùn)行到一定位置時暫停。因此,程序員在該位置處可以方便地查看變量的值、堆棧情況等,從而找出代碼的癥結(jié)所在。
在gdb中設(shè)置斷點非常簡單,只需在“b”后加入對應(yīng)的行號即可(這是最常用的方式,另外還有其他方式設(shè)置斷點),如下所示:
(gdb)b6
Breakpoint1at0x804846d:filetest.c,line6.
要注意的是,在gdb中利用行號設(shè)置斷點是指代碼運(yùn)行到對應(yīng)行之前將其停止,如上例中,代碼運(yùn)行到第6行之前暫停(并沒有運(yùn)行第6行)。
(3)查看斷點情況。
在設(shè)置完斷點之后,用戶可以鍵入“infob”來查看設(shè)置斷點情況,在gdb中可以設(shè)置多個斷點。
(gdb)infob
NumTypeDispEnbAddressWhat
1breakpointkeepy0x0804846dinmainattest.c:6
用戶在斷點鍵入“backrace”(只輸入“bt”即可)可以查到調(diào)用函數(shù)(堆棧)的情況,這個功能在程序調(diào)試之中使用非常廣泛,經(jīng)常用于排除錯誤或者監(jiān)視調(diào)用堆棧的情況。
(gdb)b19
(gdb)c
Breakpoin2,sum(m=50)attest.c:19
19printf(“Thesumof1-mis%dn”,n);
(gdb)bt
#0sum(m=50)attest.c:19 /*停在test.c的sum()函數(shù),第19行*/
#10x080483e8inmain()attest.c:6/*test.c的第6行調(diào)用sum函數(shù)*/
(4)運(yùn)行代碼。
接下來就可運(yùn)行代碼了,gdb默認(rèn)從首行開始運(yùn)行代碼,鍵入“r”(run)即可(若想從程序中指定行開始運(yùn)行,可在r后面加上行號)。
(gdb)r
Startingprogram:/root/workplace/gdb/test
Readingsymbolsfromsharedobjectreadfromtargetmemory...done.
LoadedsystemsuppliedDSOat0x5fb000
Breakpoint1,main()attest.c:6
6sum(50);
可以看到,程序運(yùn)行到斷點處就停止了。
(5)查看變量值。
在程序停止運(yùn)行之后,程序員所要做的工作是查看斷點處的相關(guān)變量值。在gdb中鍵入“p”+變量值即可,如下所示:
(gdb)pn
$1=0
(gdb)pi
$2=134518440
在此處,為什么變量“i”的值為如此奇怪的一個數(shù)字呢?原因就在于程序是在斷點設(shè)置的對應(yīng)行之前停止的,那么在此時,并沒有把“i”的數(shù)值賦為零,而只是一個隨機(jī)的數(shù)字。但變量“n”是在第4行賦值的,故在此時已經(jīng)為零。
小技巧 | gdb在顯示變量值時都會在對應(yīng)值之前加上“$N”標(biāo)記,它是當(dāng)前變量值的引用標(biāo)記,所以以后若想再次引用此變量就可以直接寫作“$N”,而無需寫冗長的變量名。 |
(6)單步運(yùn)行。
單步運(yùn)行可以使用命令“n”(next)或“s”(step),它們之間的區(qū)別在于:若有函數(shù)調(diào)用的時候,“s”會進(jìn)入該函數(shù)而“n”不會進(jìn)入該函數(shù)。因此,“s”就類似于Uisual等工具中的“stepin”,“n”類似與Uisual等工具中的“stepover”。它們的使用如下所示:
(gdb)n
Thesumof1-mis1275
7for(i=1;i=50;i++)
(gdb)s
sum(m=50)attest.c:16
16inti,n=0;
可見,使用“n”后,程序顯示函數(shù)sum()的運(yùn)行結(jié)果并向下執(zhí)行,而使用“s”后則進(jìn)入sum()函數(shù)之中單步運(yùn)行。
(7)恢復(fù)程序運(yùn)行
在查看完所需變量及堆棧情況后,就可以使用命令“c”(continue)恢復(fù)程序的正常運(yùn)行了。這時,它會把剩余還未執(zhí)行的程序執(zhí)行完,并顯示剩余程序中的執(zhí)行結(jié)果。以下是之前使用“n”命令恢復(fù)后的執(zhí)行結(jié)果:
(gdb)c
Continuing.
Thesumof1-50is:1275
Programexitedwithcode031.
可以看出,程序在運(yùn)行完后退出,之后程序處于“停止?fàn)顟B(tài)”。
小知識 | 在gdb中,程序的運(yùn)行狀態(tài)有“運(yùn)行”、“暫停”和“停止”3種,其中“暫停”狀態(tài)為程序遇到了斷點或觀察點之類的,程序暫時停止運(yùn)行,而此時函數(shù)的地址、函數(shù)參數(shù)、函數(shù)內(nèi)的局部變量都會被壓入“棧”(Stack)中。故在這種狀態(tài)下可以查看函數(shù)的變量值等各種屬性。但在函數(shù)處于“停止”狀態(tài)之后,“棧”就會自動撤消,它也就無法查看各種信息了。 |
linux相關(guān)文章:linux教程
評論