溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

C/C++程序調(diào)試和內(nèi)存檢測(cè)的方法是什么

發(fā)布時(shí)間:2021-11-30 17:55:15 來源:億速云 閱讀:97 作者:iii 欄目:移動(dòng)開發(fā)

本篇內(nèi)容主要講解“C/C++程序調(diào)試和內(nèi)存檢測(cè)的方法是什么”,感興趣的朋友不妨來看看。本文介紹的方法操作簡(jiǎn)單快捷,實(shí)用性強(qiáng)。下面就讓小編來帶大家學(xué)習(xí)“C/C++程序調(diào)試和內(nèi)存檢測(cè)的方法是什么”吧!

1、常用的調(diào)試技巧

(1)代碼檢查,重新閱讀程序,排除比較明顯的錯(cuò)誤。編譯時(shí)帶上-Wall參數(shù),生成所有的警告信息。

gcc -Wall -pedantic -ansi   表示以ansi/iso生成所有的警告西信息。

(2)取樣法:在程序中增加一些代碼,收集更多與程序運(yùn)行時(shí)的行為相關(guān)的信息。使用條件編譯,可以清楚的辨別哪些是調(diào)試代碼,有利于調(diào)試后的代碼整理。

例如:

#ifdef DEBUG
 std::cout << x :
  #endif

程序編譯時(shí)可以選擇性的加上-DDEBUG。如果加上這個(gè)標(biāo)志,就定義了DEBUG這個(gè)符號(hào),從而在程序中包含調(diào)試用的額外代碼,沒有加上該標(biāo)志,這些調(diào)試代碼將刪除。

(3)程序的受控執(zhí)行。用調(diào)試器來控制代碼的運(yùn)行,隨時(shí)查看這些變量的狀態(tài)。

為了能夠調(diào)試程序,需要在編譯和鏈接時(shí)為每個(gè)源文件加上編譯選項(xiàng)參數(shù)。這些選項(xiàng)的作用是讓編譯器在程序中添加額外的調(diào)試信息。這些信息包括符號(hào)和源代碼行號(hào),調(diào)試器將利用這些信息向用戶顯示程序已經(jīng)執(zhí)行到的源代碼的位置。-g標(biāo)志是對(duì)程序調(diào)試性編譯時(shí)常用的一個(gè)選項(xiàng)。調(diào)試信息的加入使可執(zhí)行程序的長(zhǎng)度成倍的增長(zhǎng)、容量增加,程序運(yùn)行時(shí)的內(nèi)存數(shù)量還是和原來一樣,程序調(diào)試結(jié)束后,最好還是將調(diào)試信息從程序的發(fā)行版中刪除。

2、使用gdb進(jìn)行程序調(diào)試

常用功能命令:

g++ -g -o test test.cpp  //編譯時(shí)加上-g參數(shù)

1、啟動(dòng)gdb: gdb test

2、help

3、具備帶有歷史記錄的命令行編輯功能,方向鍵選擇之前執(zhí)行過的命令,直接回車鍵再次執(zhí)行最近執(zhí)行過的那條命令。單步調(diào)試非常有用。

4、quit:退出

5、run:執(zhí)行這個(gè)程序,程序運(yùn)行失敗時(shí)gdb會(huì)報(bào)告失敗的原因和位置。

6、backtrace(bt):棧跟蹤,失敗時(shí)停止的位置,幫助我們找到程序到達(dá)錯(cuò)誤地點(diǎn)的路徑。

7、print:run 后檢查變量,注意變量的生命期。

8、打印圍繞當(dāng)前位置前后的一段代碼,繼續(xù)使用list可以顯示更多的代碼。

9、設(shè)置斷點(diǎn),停止程序的運(yùn)行,查看變量。help breakpoint,break lineNumber,cont,end,display,disable breakpoint number,clear,commands breakpointNumber.  

10、設(shè)置斷點(diǎn)后經(jīng)常使用單步調(diào)試命令next(n),查看程序運(yùn)行的細(xì)節(jié)。 

3、valgrind內(nèi)存調(diào)試

動(dòng)態(tài)內(nèi)存分配很容易出現(xiàn)程序漏洞,必須清楚自己分配的每一塊內(nèi)存,而且要確定沒有使用已經(jīng)釋放的內(nèi)存塊,非常重要。內(nèi)存調(diào)試的工具有很多,這里使用的是valgrind工具。在centos 7中直接使用 yum install valgrind 安裝。

#include <iostream>

int main()
{
    int *ptr = new int [3];
    ptr[3]=1;

    delete [] ptr;
    std::cout << ptr[1];
    return ;
}

上面簡(jiǎn)單的代碼編譯運(yùn)行不會(huì)發(fā)生錯(cuò)誤,但是實(shí)際上發(fā)生了很嚴(yán)重的內(nèi)存問題。ptr[3]訪問越界,std::cout <<ptr[i],讀已經(jīng)釋放過的內(nèi)存。

通過valgrind工具可以檢查出來:

[xgwang@localhost Desktop]$ g++ -g -o test2 test2.cpp

[xgwang@localhost Desktop]$ valgrind ./test2

==21739== Memcheck, a memory error detector
==21739== Copyright (C) 2002-2013, and GNU GPL'd, by Julian Seward et al.
==21739== Using Valgrind-3.10.0 and LibVEX; rerun with -h for copyright info
==21739== Command: ./test2
==21739== 
==21739== Invalid write of size 4
==21739== at 0x40081E: main (test2.cpp:8)
==21739== Address 0x5a1504c is 0 bytes after a block of size 12 alloc'd
==21739== at 0x4C2A7AA: operator new[](unsigned long) (in /usr/lib64/valgrind/vgpreload_memcheck-amd64-linux.so)
==21739== by 0x400811: main (test2.cpp:7)
==21739== 
==21739== Invalid read of size 4
==21739== at 0x40083F: main (test2.cpp:11)
==21739== Address 0x5a15044 is 4 bytes inside a block of size 12 free'd
==21739== at 0x4C2B5E1: operator delete[](void*) (in /usr/lib64/valgrind/vgpreload_memcheck-amd64-linux.so)
==21739== by 0x400836: main (test2.cpp:10)
==21739== 
0==21739== 
==21739== HEAP SUMMARY:
==21739== in use at exit: 0 bytes in 0 blocks
==21739== total heap usage: 1 allocs, 1 frees, 12 bytes allocated
==21739== 
==21739== All heap blocks were freed -- no leaks are possible
==21739== 
==21739== For counts of detected and suppressed errors, rerun with: -v
==21739== ERROR SUMMARY: 2 errors from 2 contexts (suppressed: 2 from 2)

到此,相信大家對(duì)“C/C++程序調(diào)試和內(nèi)存檢測(cè)的方法是什么”有了更深的了解,不妨來實(shí)際操作一番吧!這里是億速云網(wǎng)站,更多相關(guān)內(nèi)容可以進(jìn)入相關(guān)頻道進(jìn)行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

c++
AI