首页
/
每日頭條
/
科技
/
c++ 如何避免内存洩漏
c++ 如何避免内存洩漏
更新时间:2025-07-09 00:24:20
前言

寫c 的程序員都應該對申請内存和釋放内存有着深刻的領悟(可能有些初級用着前人封裝的智能指針感受不深)。同時對于出現崩潰生成可以調試的dump文件也極為重要,對于win下的發布版程序很重要。

工具

crtdbg偵測内存洩露,dbghelp 生産MINIDUMP

内存洩漏是在vs開發中,程序結束在vs輸出裡面會提示有沒有内存洩露。dump文件是在程序運行工程中崩潰時候捕捉異常後生成的,要配合對應的pdb文件和代碼用vs進行分析。

實例

樣例中的 memorytools.cpp memorytools.h 可以直接拿過去用的,不需要額外引用頭文件和庫。

代碼目錄:

├── main.cpp├── memorytools.cpp (附在文章後面面)└──memorytools.h(附在文章後面面)

1.内存洩漏檢測

main.cpp

#include "memorytools.h" int main() { //設置生成dump文件名 lge::CMemoryTools::initAppName("test"); //初始化 lge::CMemoryTools::initDebug(); //測試内存洩露 char * p = (char*) malloc(1000); int* pInt = new int; return 0; }

運行後提示如下:

c++ 如何避免内存洩漏(内存洩漏檢測和dump文件生成)1

如果申請的内存,在程序退出後還沒有釋放,會提示你内存洩漏。當然很多項目用的單例,在程序關閉的時候,不釋放一樣會提示内存洩漏。

如果想知道内存是哪一步申請的,比如我圓圈圈出的63,可以直接在CMemoryTools::initDebug函數内将_CrtSetBreakAlloc(0); 改成 _CrtSetBreakAlloc(63); 這種vs在調試的時候,就會自動斷點到這步的内存申請(但對于大項目這種方式其實不适用)。

2.dump生成及調試

#include "Memorytools.h" class CTest { public: int p; }; int main() { //設置生成dump文件名 lge::CMemoryTools::initAppName("test"); //初始化 lge::CMemoryTools::initDebug(); //測試崩潰 CTest* pTest = NULL; pTest->p = 1; return 0; }

在exe所在目錄直接雙擊打開,會看到生成的dump文件。

c++ 如何避免内存洩漏(内存洩漏檢測和dump文件生成)2

保障exe pdb 和代碼都是對應的關系,将dmp文件直接拖進vs項目即可。

c++ 如何避免内存洩漏(内存洩漏檢測和dump文件生成)3

之後就可以直接調試了。

memorytools.h

#pragma once #if WIN32 #include <Windows.h> #include <DbgHelp.h> #pragma warning(disable:4091) #pragma comment(lib,"Dbghelp.lib") #include <tchar.h> #include <stdlib.h> #include <stdio.h> #include <atlstr.h> #include <string> #ifdef DEBUG # ifdef _MSC_VER # ifndef _CrtDBG_MAP_ALLOC # define _CRTDBG_MAP_ALLOC # endif #ifndef _MAPNEW #define _MAPNEW #endif # ifdef _MAPNEW # ifndef _CRTDBG_MAP_ALLOC_NEW # define _CRTDBG_MAP_ALLOC_NEW # endif # endif # include <crtdbg.h> # ifdef _MAPNEW # ifndef new # define DEBUG_NORMALBLOCK new(_NORMAL_BLOCK, __FILE__, __LINE__) # define new DEBUG_NORMALBLOCK # endif # endif # endif #endif #ifndef __FILE_LINE__ # define _TLN(LN) #LN # define __TLINE__(LN) _TLN(LN) # define __FILE_LINE__ __FILE__"("__TLINE__(__LINE__)")" #endif #define APP_NAME_TYPE CString #else #define APP_NAME_TYPE std::string #endif namespace lge { /* * windows下内存洩漏檢測工具 */ class CMemoryTools { public: static void initAppName(const APP_NAME_TYPE& name); static void initDebug(); static APP_NAME_TYPE __app_name__; static bool _initDebug; }; }

memorytools.cpp

#include "memorytools.h" APP_NAME_TYPE lge::CMemoryTools::__app_name__ = "engine"; bool lge::CMemoryTools::_initDebug = false; #if WIN32 int MemoryLeakReportRoutine(int blockType, char *message, int *p) { if (strcmp(message, "Object dump complete.\n") == 0) { static const char * szSolvMemLeak = "請解決内存洩露!\r\n"; OutputDebugStringA(szSolvMemLeak); } return 0; } LONG WINAPI UnHANDLEExceptionFilte(EXCEPTION_POINTERS *ExceptionInfo) { SYSTEMTIME Systime; GetLocalTime(&Systime); CString Str; Str.Format(_T("%s_%d-%d-%d-d-d-d.dmp"), lge::CMemoryTools::__app_name__, Systime.wYear, Systime.wMonth, Systime.wDay, Systime.wHour, Systime.wMinute, Systime.wSecond); HANDLE hFile = CreateFile(Str, GENERIC_READ | GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0); MiniDump_TYPE wDumpFlag = MiniDumpWithFullMemory; if (hFile != INVALID_HANDLE_VALUE) { MINIDUMP_EXCEPTION_INFORMATION ExInfo; ExInfo.ThreadId = GetCurrentThreadId(); ExInfo.ExceptionPointers = ExceptionInfo; ExInfo.ClientPointers = NULL; MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hFile, wDumpFlag, &ExInfo, NULL, NULL); CloseHandle(hFile); } ExitProcess(-1); return 0; } #endif //WIN32 void lge::CMemoryTools::initAppName(const APP_NAME_TYPE& name) { __app_name__ = name; } void lge::CMemoryTools::initDebug() { if (_initDebug) { return; } _initDebug = true; #ifdef WIN32 #ifdef DEBUG _set_error_mode(_OUT_TO_MSGBOX); _CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF); _CrtSetReportMode(_CRT_ERROR, _CRTDBG_MODE_DEBUG); _CrtSetReportHook(&MemoryLeakReportRoutine); _CrtSetBreakAlloc(0); #endif // DEBUG SetUnhandledExceptionFilter(UnhandleExceptionFilte); #endif }

,
Comments
Welcome to tft每日頭條 comments! Please keep conversations courteous and on-topic. To fosterproductive and respectful conversations, you may see comments from our Community Managers.
Sign up to post
Sort by
Show More Comments
推荐阅读
冬天柴油汽車難啟動怎麼辦
冬天柴油汽車難啟動怎麼辦
1、氣溫低季節應進行預熱柴油發起機在冷啟動時,因不能到達柴油的壓燃溫度是影響啟動性能的主要起因。對此,可将熱水參加發起機冷卻系預熱,這是改良啟動性能的有效門路。2、詳細做法是:延續加熱水(讓水流出缸體),并逐步進步水溫進行預熱。當流出的水溫度較高時再封閉放水開關。此外,可用噴燈等明火對油底殼進行加熱...
2025-07-09
qq發紅包步驟
qq發紅包步驟
1、qq發紅包步驟:登錄QQ點擊頭像,然後點擊QQ錢包;然後點擊發紅包;我們可以選擇發普通紅包或者拼...
2025-07-09
新生兒黃疸怎麼退黃快
新生兒黃疸怎麼退黃快
第一、退黃疸最快的方法是去醫院照藍光,最好黃疸剛出現就照,一般照5-7天就消退,但是嬰兒要與寶媽隔離,對嬰兒和寶媽來說,都不好,黃疸不嚴重的話,一般不建議照藍光。第二、黃疸不嚴重的情況下可以帶孩子出去曬曬太陽,在陽光的照射下,黃疸會消得快些,但是要避免陽光照射到嬰兒的眼睛,和注意嬰兒的保暖。第三、盡量給寶寶喂母乳,寶寶喝母乳,母乳好吸收,且清淡不上火,寶寶喝母乳排洩次數多,黃疸中的毒素會随着排洩物
2025-07-09
wifi信号滿格但網速慢怎麼解決
wifi信号滿格但網速慢怎麼解決
1、嘗試更改WiFi的通道。很多路由器的WiFi通道默認是8或者自動,将通道重新更換後可以刷新WiFi的傳遞速度。2、更改路由器的頻率。路由器的傳遞頻率一般為5赫茲和2.4赫茲,5赫茲的頻率傳輸速度快,穿透力弱,2.4赫茲的頻率傳遞慢,但是穿透力很強,可以更換新的WiFi頻率嘗試。3、查看路由器的IP是否穩定。如果路由器信号才用的是DHCP的IP格式,可以更換成靜态IP嘗試。
2025-07-09
紅米手機通話錄音在哪裡找
紅米手機通話錄音在哪裡找
演示機型:紅米k40系統版本:MIUI121、打開MIUI系統中的“文件管理”應用APP。2、點擊頁面頂部“手機”選項卡,進入MIUI文件夾目錄。3、找到sound_recorder文件夾目錄。4、在該文件夾目錄下即可查看到保存的錄音文件。紅米手機小技巧:1、遠程協助:這個功能主要是方便家裡的父母長輩,當他們不會操作的時候兩台小米手機之間進行連接,連
2025-07-09
Copyright 2023-2025 - www.tftnews.com All Rights Reserved