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

寫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
推荐阅读
汽車中控屏怎麼鍊接電腦
汽車中控屏怎麼鍊接電腦
汽車中控屏怎麼鍊接電腦?使用手機的原裝USB數據線将手機與車機連接并配對,今天小編就來說說關于汽車中控屏怎麼鍊接電腦?下面更多詳細答案一起來看看吧!汽車中控屏怎麼鍊接電腦使用手機的原裝USB數據線将手機與車機連接并配對。手機連接後,自動彈出...
2025-09-13
千元機哪款手機性價比高值得購買
千元機哪款手機性價比高值得購買
榮耀暢玩5X雙卡雙待手機采用全金屬機身打造,配備5.5吋FHD1080P全高清黑瀑布屏。顔色方面提供了落日金,暗夜灰,破曉銀3種機身顔色可供用戶選擇。配置方面,榮耀暢玩5X搭載了骁龍615八核處理器。内存方面,全網通版本采用3GBRAM16...
2025-09-13
微信PC端登錄和手機端登錄邏輯分享
微信PC端登錄和手機端登錄邏輯分享
微信PC端登錄和手機端登錄邏輯分享?PC端登錄需要在微信開放平台申請,而移動端授權登錄隻需要有已經認證的服務号即可然後同一個網站平台,用同一個微信用戶去登錄,會發現PC端獲取的微信用戶openid和移動端不同,這個問題如何解決?,我來為大家...
2025-09-13
星露谷物語win10系統停止運行問題解決方法
星露谷物語win10系統停止運行問題解決方法
輸入法可以不用删,運行遊戲,進入全屏後,在這幾秒鐘内(最遲讀檔之前)把輸入法切換成英,讀檔之後再切換就不行了,然後,就可以開心的玩了。需要注意的是:遊戲時可以彈出去浏覽網頁,挂着都可以,但是不能切換輸入法,切了又會停止運行。星露谷物語專題S...
2025-09-13
amd6600顯卡發布會
amd6600顯卡發布會
目前在京東商城上,AMD四款RX6000系列顯卡已跌破首發價,價格都是參考的AIB品牌旗艦店的價格。這四款顯卡分别為RX6900XT、RX6600XT、RX6600、RX6500XT。RX6900XT顯卡首發價7999元。目前藍寶石的超白金...
2025-09-13
Copyright 2023-2025 - www.tftnews.com All Rights Reserved