VC++ 各种未处理异常的处理,并输出DUMP崩溃转储调试文件

2024-05-11 13:44

本文主要是介绍VC++ 各种未处理异常的处理,并输出DUMP崩溃转储调试文件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文包含:

信号处理器、STL标准库异常处理、STL未处理器异常处理、SEH结构化未处理异常处理、C++ 未定义虚函数异常处理、C/C++ 内存分配异常处理等。

设置异常处理器:

            // Windows platforms need to mount unhandled exception handlers so that they can print dump debug files for app crashes.
#if defined(_DEBUG)::_CrtSetDbgFlag(_CRTDBG_LEAK_CHECK_DF | _CRTDBG_ALLOC_MEM_DF);
#endif::_set_abort_behavior(_CALL_REPORTFAULT, _CALL_REPORTFAULT);::_set_purecall_handler(Crt_HandlePureVirtualCall);::_set_new_handler(Crt_NewHandler); /* std::set_new_handler(...) */#if _MSC_VER >= 1400 ::_set_invalid_parameter_handler(Crt_InvalidParameterHandler);
#endif::signal(SIGABRT, Crt_SigabrtHandler);::signal(SIGINT, Crt_SigabrtHandler);::signal(SIGTERM, Crt_SigabrtHandler);::signal(SIGILL, Crt_SigabrtHandler);::set_terminate(Crt_TerminateHandler);::set_unexpected(Crt_UnexpectedHandler);::SetUnhandledExceptionFilter(Seh_UnhandledExceptionFilter);

异常处理器实现:

        static ppp::string Seh_NewDumpFileName() noexcept{ppp::string path = ppp::GetExecutionFileName();std::size_t index = path.rfind(".");if (index != ppp::string::npos){path = path.substr(0, index);}struct tm tm_;time_t datetime = time(NULL);localtime_s(&tm_, &datetime);char sz[1000];sprintf_s(sz, sizeof(sz), "%04d%02d%02d-%02d%02d%02d", 1900 + tm_.tm_year, 1 + tm_.tm_mon, tm_.tm_mday, tm_.tm_hour, tm_.tm_min, tm_.tm_sec);path = path + "-" + sz + ".dmp";path = "./" + path;path = ppp::io::File::RewritePath(path.data());path = ppp::io::File::GetFullPath(path.data());return path;}static LONG WINAPI Seh_UnhandledExceptionFilter(EXCEPTION_POINTERS* exceptionInfo) noexcept{// Give user code a chance to approve or prevent writing a minidump.  If the// filter returns false, don't handle the exception at all.  If this method// was called as a result of an exception, returning false will cause// HandleException to call any previous handler or return// EXCEPTION_CONTINUE_SEARCH on the exception thread, allowing it to appear// as though this handler were not present at all.HANDLE hFile = CreateFileA(Seh_NewDumpFileName().data(), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);if (hFile != INVALID_HANDLE_VALUE){MINIDUMP_EXCEPTION_INFORMATION exceptionParam;exceptionParam.ThreadId = GetCurrentThreadId();exceptionParam.ExceptionPointers = exceptionInfo;exceptionParam.ClientPointers = TRUE;MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), hFile, MiniDumpWithFullMemory, &exceptionParam, NULL, NULL);CloseHandle(hFile);}// The handler either took care of the invalid parameter problem itself,// or passed it on to another handler.  "Swallow" it by exiting, paralleling// the behavior of "swallowing" exceptions.exit(-1); /* abort(); */return EXCEPTION_EXECUTE_HANDLER;}#if _MSC_VER >= 1400 // https://chromium.googlesource.com/breakpad/breakpad/src/+/master/client/windows/handler/exception_handler.ccstatic void __CRTDECL Crt_InvalidParameterHandler(const wchar_t* expression, const wchar_t* function, const wchar_t* file, unsigned int line, uintptr_t pReserved) noexcept{std::wcerr << L"Invalid parameter detected:" << std::endl;std::wcerr << L"Expression: " << expression << std::endl;std::wcerr << L"Function: " << function << std::endl;std::wcerr << L"File: " << file << std::endl;std::wcerr << L"Line: " << line << std::endl;_CrtDumpMemoryLeaks();_CrtSetReportMode(_CRT_ASSERT, _CRTDBG_MODE_DEBUG);_CrtMemDumpAllObjectsSince(NULL);_CrtSetReportFile(_CRT_ASSERT, _CRTDBG_FILE_STDERR);// Make up an exception record for the current thread and CPU context// to make it possible for the crash processor to classify these// as do regular crashes, and to make it humane for developers to// analyze them.EXCEPTION_RECORD exception_record = {};CONTEXT exception_context = {};EXCEPTION_POINTERS exception_ptrs = { &exception_record, &exception_context };::RtlCaptureContext(&exception_context);exception_record.ExceptionCode = STATUS_INVALID_PARAMETER;// We store pointers to the the expression and function strings,// and the line as exception parameters to make them easy to// access by the developer on the far side.exception_record.NumberParameters = 4;exception_record.ExceptionInformation[0] = reinterpret_cast<ULONG_PTR>(expression);exception_record.ExceptionInformation[1] = reinterpret_cast<ULONG_PTR>(file);exception_record.ExceptionInformation[2] = line;exception_record.ExceptionInformation[3] = reinterpret_cast<ULONG_PTR>(function);// Deliver exceptions to unhandled exception handler.Seh_UnhandledExceptionFilter(&exception_ptrs);}
#endifstatic int Seh_NoncontinuableException() noexcept{// Make up an exception record for the current thread and CPU context// to make it possible for the crash processor to classify these// as do regular crashes, and to make it humane for developers to// analyze them.EXCEPTION_RECORD exception_record = {};CONTEXT exception_context = {};EXCEPTION_POINTERS exception_ptrs = { &exception_record, &exception_context };::RtlCaptureContext(&exception_context);exception_record.ExceptionCode = STATUS_NONCONTINUABLE_EXCEPTION;// We store pointers to the the expression and function strings,// and the line as exception parameters to make them easy to// access by the developer on the far side.exception_record.NumberParameters = 3;exception_record.ExceptionInformation[0] = NULL;exception_record.ExceptionInformation[1] = NULL;exception_record.ExceptionInformation[2] = 0;// Deliver exceptions to unhandled exception handler.return Seh_UnhandledExceptionFilter(&exception_ptrs);}static int __CRTDECL Crt_NewHandler(size_t) noexcept{return Seh_NoncontinuableException();}static void __CRTDECL Crt_HandlePureVirtualCall() noexcept{Seh_NoncontinuableException();}static void __CRTDECL Crt_TerminateHandler() noexcept{Seh_NoncontinuableException();}static void __CRTDECL Crt_UnexpectedHandler() noexcept{Seh_NoncontinuableException();}static void __CRTDECL Crt_SigabrtHandler(int) noexcept{Seh_NoncontinuableException();}

这篇关于VC++ 各种未处理异常的处理,并输出DUMP崩溃转储调试文件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/979699

相关文章

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

PHP轻松处理千万行数据的方法详解

《PHP轻松处理千万行数据的方法详解》说到处理大数据集,PHP通常不是第一个想到的语言,但如果你曾经需要处理数百万行数据而不让服务器崩溃或内存耗尽,你就会知道PHP用对了工具有多强大,下面小编就... 目录问题的本质php 中的数据流处理:为什么必不可少生成器:内存高效的迭代方式流量控制:避免系统过载一次性

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Python实现批量CSV转Excel的高性能处理方案

《Python实现批量CSV转Excel的高性能处理方案》在日常办公中,我们经常需要将CSV格式的数据转换为Excel文件,本文将介绍一个基于Python的高性能解决方案,感兴趣的小伙伴可以跟随小编一... 目录一、场景需求二、技术方案三、核心代码四、批量处理方案五、性能优化六、使用示例完整代码七、小结一、

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

MyBatis-plus处理存储json数据过程

《MyBatis-plus处理存储json数据过程》文章介绍MyBatis-Plus3.4.21处理对象与集合的差异:对象可用内置Handler配合autoResultMap,集合需自定义处理器继承F... 目录1、如果是对象2、如果需要转换的是List集合总结对象和集合分两种情况处理,目前我用的MP的版本