内存不足引发C++程序闪退崩溃问题的分析与总结
目录
1、内存不足一般出现在32位程序中
2、内存不足时会导致malloc或new申请内存失败
2.1、malloc申请内存失败,返回NULL
2.2、new申请内存失败,抛出异常
3、内存不足项目实战案例中相关细节与要点说明
3.1、内存不足导致malloc申请内存失败,导致程序闪退
3.1.1、内存不足导致程序闪退的原因
3.1.2、abort强制终止进程导致程序闪退时,不会生成dump文件
3.2、内存不足导致new申请内存时抛出异常,引发程序崩溃
4、引发内存不足的两个原因
4.1、32位程序默认的用户态虚拟内存只有2GB,程序模块较多,可能占用的内存接近2GB了
4.2、程序中有内存泄漏,且内存泄漏的代码在不断的执行,导致程序占用内存越来越多
5、解决内存不够用的办法
6、最后
C++软件异常排查从入门到精通系列教程(专栏文章列表,欢迎订阅,持续更新...)https://blog.csdn.net/chenlycly/article/details/125529931C/C++实战进阶(已更新到460多篇,持续更新中...)
https://blog.csdn.net/chenlycly/article/details/140824370VC++常用功能开发汇总(专栏文章列表,欢迎订阅,持续更新...)
https://blog.csdn.net/chenlycly/article/details/124272585Windows C++ 软件开发从入门到精通(专栏文章,持续更新中...)
https://blog.csdn.net/chenlycly/category_12695902.htmlC++软件分析工具从入门到精通案例集锦(专栏文章,持续更新中...)
https://blog.csdn.net/chenlycly/article/details/131405795开源组件及数据库技术(专栏文章,持续更新中...)
https://blog.csdn.net/chenlycly/category_12458859.html网络编程与网络问题分享(专栏文章,持续更新中...)
https://blog.csdn.net/chenlycly/category_2276111.html 最近在项目中遇到几个因为内存不足引发程序闪退或崩溃的问题,比较有代表性,所以本文在此总结一下内存不足的相关问题场景以及内存不足引发程序异常的具体原因,供大家借鉴或参考。
1、内存不足一般出现在32位程序中
C++程序按位数可以分为32位和64位程序。对于64位程序,内存地址按64位寻址,系统会给64位程序分配0-0xFFFFFFFFFFFFFFFF这个范围的虚拟地址。而对于32位程序,内存地址按32位寻址,系统会给32位程序分配0-0xFFFFFFFF这个范围的虚拟内存,即4GB虚拟内存。所以系统给64位程序分配的虚拟内存空间要比32位程序大的多,几乎是“用之不竭”的,所以内存不足基本是出现在32位程序中。
对于32位程序,系统在其启动时分配4GB的虚拟内存,默认情况下,用户态虚拟内存和内核态虚拟内存各占2GB,而C++程序基本都运行在用户态,用户态虚拟内存只有2GB,当占用内存的模块很多或者有内存泄漏时,很容易接近这个2GB的上限,从而出现内存不足的问题,进而导致程序发生闪退或崩溃。
虽然系统给64程序分配的虚拟内存非常大,但在有内存泄漏的情况下也可能会出现内存不足的问题。比如程序一直在持续的运行(程序不关闭),且在持续的发生内存泄漏,则再大的虚拟内存空间也有用尽的那一刻。
2、内存不足时会导致malloc或new申请内存失败
C++程序在运行过程中主要使用全局内存、栈内存和堆内存,而全局内存是在程序启动时(进入main函数之前)就分配好了,栈内存和堆内存则是在程序运行过程中分配或申请的。栈内存则是和线程相关的,在创建线程时系统会给线程分配指定大小的栈内存,线程中调用的函数中的局部变量使用的都是栈内存。
在Windows系统中,系统默认给线程分配1MB的栈空间;在Linux系统中,系统默认给线程分配8MB的栈空间。线程中调用的函数会使用线程的栈空间,如果当前线程中调用的函数占用的总的栈空间超过系统分配给系统的上限,则会导致Stack oveflow线程栈溢出的异常,导致程序崩溃。关于引发Stack oveflow线程栈溢出的常见原因及详细说明,可以查看我的文章:
通过Stack Overflow线程栈溢出的问题实例,详解C++程序线程栈溢出的诸多细节
https://blog.csdn.net/chenlycly/article/details/140908516
线程占用的栈空间相对不多,C++程序中使用更多的还是堆内存,所以我们此处主要讨论内存不足对申请堆内存的影响。C++程序中主要使用C函数malloc或者new操作符去申请堆内存,当内存不足时使用malloc和new的反应是有所不同的:
1)malloc申请内存失败时会返回NULL;
2)new申请内存失败时会抛出异常,导致程序发生崩溃。
在Windows C++程序中,除了使用new去申请堆内存,还可以调用Windows系统API函数HeapAlloc从堆上分配内存(要用HeapFree去释放),还可能调用API函数VirtualAlloc或VirtualAllocEx从虚拟内存上分配内存(要用VirtualFree或VirtualFreeEx去释放),还可能调用其他的API函数。本文主要讨论使用new的场景。
2.1、malloc申请内存失败,返回NULL
malloc申请内存失败时,会返回NULL,通过这个NULL返回值就可以判断内存申请失败了。
malloc申请内存失败,可能会引发程序发生闪退(注意内存申请失败本身不会导致程序闪退崩溃)。之前我们在项目中就遇到过,我们的程序使用了开源的WebRTC库,当时程序中有内存泄漏,内存不够用了,导致WebRTC开源库内部代码在调用malloc申请内存时失败了,malloc返回NULL,WebRTC内部判断malloc返回NULL,认为这个是fatal致命的(因为申请不到内存,业务没法正常的展开,WebRTC内部认为这是致命的),直接调用abort强行将当前进程终止了,相关代码如下:
RTC_CHECK宏会校验malloc的返回值,如下:
如果malloc返回的地址为空,则会调用rtc_FatalMessage接口:
rtc_FatalMessage接口内部会调用FatalLog:
FatalLog接口中会在末尾处先是调用DebugBreak接口,如果当前程序正在调试,则DebugBreak会让调试器中断下来(从DebugBreak函数名就能看出这个函数的作用),目的是让正在调试的用户感知到!然后紧接着会调用abort接口,将当前进程强制终止掉!
其实abort接口内部也会让正在调试的调试器中断下来,abort的内部实现代码如下所示:
/***
*void abort() - abort the current program by raising SIGABRT
*
*Purpose:
* print out an abort message and raise the SIGABRT signal. If the user
* hasn't defined an abort handler routine, terminate the program
* with exit status of 3 without cleaning up.
*
* Multi-thread version does not raise SIGABRT -- this isn't supported
* under multi-thread.
*******************************************************************************/
void __cdecl abort (void)
{_PHNDLR sigabrt_act = SIG_DFL;#ifdef _DEBUGif (__abort_behavior & _WRITE_ABORT_MSG){/* write the abort message */_NMSG_WRITE(_RT_ABORT);}
#endif /* _DEBUG *//* Check if the user installed a handler for SIGABRT.* We need to read the user handler atomically in the case* another thread is aborting while we change the signal* handler.*/sigabrt_act = __get_sigabrt();if (sigabrt_act != SIG_DFL){raise(SIGABRT);}/* If there is no user handler for SIGABRT or if the user* handler returns, then exit from the program anyway*/if (__abort_behavior & _CALL_REPORTFAULT){_call_reportfault(_CRT_DEBUGGER_ABORT, STATUS_FATAL_APP_EXIT, EXCEPTION_NONCONTINUABLE);}/* If we don't want to call ReportFault, then we call _exit(3), which is the* same as invoking the default handler for SIGABRT*/_exit(3);
}
从上述代码可以看出,abort内部调用了raise(SIGABRT),该函数是触发一个SIGABRT信号终止异常,如果当前正在调试状态,会让调试器中断下来。最后调用了C函数_exit退出当前进程。
注意,上述问题中并没有产生C++异常,是调用abort强行终止进程的,程序直接闪退消失了,给人感觉是程序崩溃了,实际上并不是崩溃!这个项目问题案例,我之前专门写了文章,可以去查看文章:
WebRTC开源库内部调用abort函数引发C++程序发生闪退问题的详细排查https://blog.csdn.net/chenlycly/article/details/129460580 对于这种调用abort强行终止进程的问题,还有一个细节值得注意一下。如果程序中安装了异常捕获,此种场景下是不会生成dump文件的。因为当前是WebRTC开源库内部检测到malloc返回NULL,直接调用abort,程序并没有发生C++上的异常,所以异常捕获是感知不到的(发生C++上的异常,异常捕获模块才能感知到),虽然给人一种程序发生崩溃闪退的感觉,但不会生成dump文件的。
在这里,给大家重点推荐一下我的几个热门畅销专栏,欢迎订阅:(博客主页还有其他专栏,可以去查看)
专栏1:(该精品技术专栏的订阅量已达到560多个,专栏中包含大量项目实战分析案例,有很强的实战参考价值,广受好评!专栏文章持续更新中,预计更新到200篇以上!欢迎订阅!)
C++软件调试与异常排查从入门到精通系列文章汇总https://blog.csdn.net/chenlycly/article/details/125529931
本专栏根据多年C++软件异常排查的项目实践,系统地总结了引发C++软件异常的常见原因以及排查C++软件异常的常用思路与方法,详细讲述了C++软件的调试方法与手段,以图文并茂的方式给出具体的项目问题实战分析实例(很有实战参考价值),带领大家逐步掌握C++软件调试与异常排查的相关技术,适合基础进阶和想做技术提升的相关C++开发人员!
考察一个开发人员的水平,一是看其编码及设计能力,二是要看其软件调试能力!所以软件调试能力(排查软件异常的能力)很重要,必须重视起来!能解决一般人解决不了的问题,既能提升个人能力及价值,也能体现对团队及公司的贡献!
专栏中的文章都是通过项目实战总结出来的,包含大量项目问题实战分析案例,有很强的实战参考价值!专栏文章还在持续更新中,预计文章篇数能更新到200篇以上!
专栏2:(本专栏涵盖了C++多方面的内容,是当前重点打造的专栏,订阅量已达210多个,专栏文章已经更新到470多篇,持续更新中...)
C/C++实战进阶(专栏文章,持续更新中...)https://blog.csdn.net/chenlycly/category_11931267.html
以多年的开发实战为基础,总结并讲解一些的C/C++基础与项目实战进阶内容,以图文并茂的方式对相关知识点进行详细地展开与阐述!专栏涉及了C/C++领域多个方面的内容,包括C++基础及编程要点(模版泛型编程、STL容器及算法函数的使用等)、数据结构与算法、C++11及以上新特性(不仅看开源代码会用到,日常编码中也会用到部分新特性,面试时也会涉及到)、常用C++开源库的介绍与使用、代码分享(调用系统API、使用开源库)、常用编程技术(动态库、多线程、多进程、数据库及网络编程等)、软件UI编程(Win32/duilib/QT/MFC)、C++软件调试技术(排查软件异常的手段与方法、分析C++软件异常的基础知识、常用软件分析工具使用、实战问题分析案例等)、设计模式、网络基础知识与网络问题分析进阶内容等。
专栏3:
C++常用软件分析工具从入门到精通案例集锦汇总(专栏文章,持续更新中...)https://blog.csdn.net/chenlycly/article/details/131405795
常用的C++软件辅助分析工具有SPY++、PE工具、Dependency Walker、GDIView、Process Explorer、Process Monitor、API Monitor、Clumsy、Windbg、IDA Pro等,本专栏详细介绍如何使用这些工具去巧妙地分析和解决日常工作中遇到的问题,很有实战参考价值!
专栏4:
VC++常用功能开发汇总(专栏文章,持续更新中...)https://blog.csdn.net/chenlycly/article/details/124272585
将10多年C++开发实践中常用的功能,以高质量的代码展现出来。这些常用的高质量规范代码,可以直接拿到项目中使用,能有效地解决软件开发过程中遇到的问题。
专栏5:
C++ 软件开发从入门到精通(专栏文章,持续更新中...)https://blog.csdn.net/chenlycly/category_12695902.html
根据多年C++软件开发实践,详细地总结了C/C++软件开发相关技术实现细节,分享了大量的实战案例,很有实战参考价值。
2.2、new申请内存失败,抛出异常
当程序内存不足时,使用new去动态申请内存失败,默认情况下会抛出bad_alloc内存申请失败的异常。这种场景是产生了异常,如果程序中安装了异常捕获模块,且异常捕获模块感知到了,会生成dump文件的。比如我们有次遇到因为内存泄漏导致内存不足导致new抛出了异常,用Windbg打开dump文件,在函数调用堆栈中看到了程序中抛出了bad_alloc异常,如下所示:
从堆栈上可以看出,抛出bad_alloc异常的代码就是new那句代码抛出的是std::bad_alloc异常,就是我们讲的bad_alloc异常,导致程序发生崩溃闪退(代码中没有对这个bad_alloc异常进行处理)。
如果dump文件中显示new时抛出了bad_alloc异常,可能是内存不足导致的,可以怀疑程序中可能存在内存泄漏,然后使用工具观察程序运行过程中的内存变化,去确定是否真的是内存泄漏导致的!
这里还有一个细节,new失败抛出bad_alloc异常,不一定是内存不足导致的。也可能是程序中有堆内存越界,导致堆内存被破坏引发的。堆内存被破坏,会导致程序导出胡乱的崩溃,一会再new的地方,一会在delete的地方。堆内存被破坏的异常,是最难查的一类异常!
对于new失败时抛出bad_alloc异常导致程序崩溃,我们可以让new不要抛出异常,而是像malloc一样在申请内存失败时返回空指针,使用std::nothrow指定new不要抛出异常,如下:
#include <iostream>int main(){char *p = NULL;int i = 0;do{p = new(std::nothrow) char[10*1024*1024]; // 每次申请10MBi++;Sleep(5);}while(p);if(NULL == p){std::cout << "分配了 " << (i-1)*10 << " M内存" //分配了 1890 Mn内存第 1891 次内存分配失败 << "第 " << i << " 次内存分配失败";}return 0;
}
此外,还可以使用try...catch来捕获异常:(因为异常被捕获处理了,程序也就不会再崩溃了)
#include <iostream>
using namespace std;int main(){char *p;int i = 0;try{do{p = new char[10*1024*1024];i++;Sleep(5);}while(p);}catch(const std::exception& e){std::cout << e.what() << "\n"<< "分配了" << i*10 << "M" << std::endl;}return 0;
}
虽然上面两种做法,都能让程序不再崩溃,但程序进程内存不足,业务已经没法正常展开和执行了,让程序还活着,意义也不大了。
3、内存不足项目实战案例中相关细节与要点说明
最近遇到的两个问题案例都是内存不足导致的,一个是程序占用的虚拟内存接近进程的用户态虚拟内存上限导致的,一个是程序中发生了内存泄漏引发的。这两个案例中,一个是内存不足导致了WebRTC内部malloc申请内存失败导致程序发生闪退,一个是内存不足导致了new内存时抛出了异常引发了程序崩溃。下面讲一下这两个案例涉及的细节和要点。
3.1、内存不足导致malloc申请内存失败,导致程序闪退
3.1.1、内存不足导致程序闪退的原因
我们32位程序占用了大量的虚拟内存,可能快接近程序的2GB用户态虚拟内存的上限,从而导致WebRTC库内部使用malloc申请内存时出现失败的问题!WebRTC内部可能认为malloc申请内存失败是Fatal致命的,因为相关业务无法正常展开了。然后会直接调用abort接口强行将当前进程终止掉,这样程序就闪退了。具体的代码及逻辑说明已经在上面的2.1节中详细讲述了,这里就不再展开了。
3.1.2、abort强制终止进程导致程序闪退时,不会生成dump文件
调用abort强行终止进程,程序闪退,给人一种程序发生崩溃的感觉,其实程序中并没有产生C++异常,并没有产生崩溃,所以程序中安装的异常捕获模块没有感知到(只有发生异常才会感知到),所以就没有生成dump文件。所以,项目中如果遇到程序闪退时没有产生dump文件,可能是调用abort函数强行终止进程引起的。
有时我们会尝试到系统的应用程序日志中查找程序崩溃的日志,因为日志中可能会记录程序AppCrash的记录,记录中可能会有dump文件的信息(包括dump文件的路径)。系统在感知到程序异常崩溃时,可能会自动生成包含异常上下文的dump文件。之前在项目中遇到的崩溃,到应用程序日志中查看到dump文件的信息。对于那些
所以,当程序发生闪退且没有生成dump文件时,可能是内存不足导致内存申请失败,程序中调用了abort强行终止进程导致的。对于这类问题,如果好复现,可以将Windbg附加到进程上进行动态调试,abort的调用会让Windbg中断下来,然后去查看此刻的函数调用堆栈就可以找出原因了,对应的案例,可以查看我的文章:
基于WebRTC构建的C++程序因虚拟内存不足导致闪退问题的排查以及解决办法的探究https://blog.csdn.net/chenlycly/article/details/133973572
3.2、内存不足导致new申请内存时抛出异常,引发程序崩溃
内存不足导致new申请内存失败,抛出异常,如果程序中安装了异常捕获模块,异常捕获模块就会感知到程序发生了异常。以我们项目遇到的一个问题为例,程序崩溃后取来了dump文件,用Windbg打开,查看异常发生的函数调用堆栈:
就是new时抛出了bad_alloc的异常,就是内存不足引起的。是因为程序中发生了内存泄漏,导致的内存不足。
对于引发C++程序内存泄漏的原因分析与排查方法,之前做过详细的总结,可以查看我之前写的文章:
引发C++程序内存泄漏的原因分析与排查方法总结https://blog.csdn.net/chenlycly/article/details/141403867 内存泄漏问题,有时排查起来会很费劲,可以借助一些工具去排查,比如使用Windbg、VLD、Debug Diagnostic Tool、Valgrind和AddressSanitizer等。
关于VLD检测内存泄漏的完整案例,可以查看我的文章:
使用Visual Leak Detector(VLD)排查C++程序内存泄漏https://blog.csdn.net/chenlycly/article/details/135472681 关于Debug Diagnostic Tool检测内存泄漏的案例,可以查看我的文章:
使用Debug Diagnostic Tool工具排查C++程序内存泄漏问题https://blog.csdn.net/chenlycly/article/details/80075888 关于Windbg检测内存泄漏的案例,可以查看我的文章:
使用Windbg排查C++程序内存泄漏问题https://blog.csdn.net/chenlycly/article/details/121295720使用 Windbg 的 !heap 命令分析内存泄漏
https://blog.csdn.net/chenlycly/article/details/131576063 内存泄漏点的定位,还可以使用历史版本比对法,相关方法可以查看我之前写的一篇类似的文章:
使用历史版本比对法排查C++程序中的内存泄漏问题https://blog.csdn.net/chenlycly/article/details/141002375此处的历史版本比对法,是排查C++软件异常的一个重要方法,之前对排查C++软件异常的常用手段与方法进行过详细的总结,可以查看我之前写的文章:
排查C++软件异常的常见思路与方法(实战经验总结)https://blog.csdn.net/chenlycly/article/details/120629327 关于Linux系统上的Valgrind和AddressSanitizer工具,可以查看我的文章:(微软从Visual Stdio 2019 19.6版本开始引入Google的AddressSanitizer内存检测工具,下面的文章有讲到)
Windows和Linux下排查C++软件异常的常用调试器与内存检测工具详细介绍https://blog.csdn.net/chenlycly/article/details/126381865为什么选择C/C++内存检测工具AddressSanitizer?如何使用AddressSanitizer?
https://blog.csdn.net/chenlycly/article/details/132863447 有时使用一个工具可能无法定位内存泄漏问题,可能需要尝试使用多个内存检测工具去排查,之前记录了内存泄漏排查的专题,感兴趣也可以去看看:
C++内存泄漏排查https://blog.csdn.net/chenlycly/category_12370029.html
4、引发内存不足的两个原因
导致程序产生内存不足,从项目中遇到的问题案例来看,主要有两种原因:
1)程序占用了大量的虚拟内存,接近给程序进程分配的虚拟内存的上限。
2)程序中发生了内存泄漏,导致程序占用的虚拟内存越来越大,接近或达到给程序进程分配的虚拟内存上限。
4.1、32位程序默认的用户态虚拟内存只有2GB,程序模块较多,可能占用的内存接近2GB了
32位程序启动时,系统会给程序分配4GB的虚拟内存,用户态和内核态默认各占2GB。如果程序中的模块比较多,或者引入了消耗内存很大的模块后,会导致主程序占用的内存很大,甚至接近用户态2GB虚拟内存的上限(程序的业务模块基本都运行在用户态中,占用的都是用户态的虚拟内存)。再执行比较耗内存的操作时,可能会导致更接近或达到2GB用户态虚拟内存的上限,导致内存不足,进而导致程序出问题。
规避的办法是,修改主程序的工程配置,启动大地址模式,将用户态虚拟内存从默认的2GB上调到3GB,这样可以有效地缓解内存不够用的问题。关于如何在Visual Studio中配置大地址模式,可以查看我之前写的文章:
如何配置32位C++程序启用大地址模式(将用户态虚拟内存从2GB扩充到3GB),以解决用户态虚拟内存不够用问题?(项目实战案例解析)https://blog.csdn.net/chenlycly/article/details/138460583
4.2、程序中有内存泄漏,且内存泄漏的代码在不断的执行,导致程序占用内存越来越多
程序中有内存泄漏,内存泄漏的代码频繁地被执行,或者用户多次执行内存泄漏的代码(也可能是测试人员进行拷机测试,比如不断的入会退会,前端时间我们就遇到过这个问题场景),导致程序占用的内存越来越多,最后导致内存不足。对于内存泄漏,可以使用Process Explorer工具持续地查看程序虚拟内存的变化,确定是否存在内存泄漏。找到内存泄漏的点并加以解决,就可以解决这类问题。
关于内存泄漏的排查,可以用历史版本比对法,也可以使用一些工具进行排查,上面已经讲到了,我就不再赘述了。
5、解决内存不够用的办法
我们先要确定引发内存不足的原因,然后针对原因和场景,提出对应的解决办法。如果程序中有内存泄漏,则要解决内存泄漏。如果程序是32位的,确实包含多个模块,占用的内存比较大,则有以下几个方法:
1)优化程序内存占用
对程序中的内存占用进行优化,减小内存的占用。
2)扩充用户态虚拟内存可以修改主工程的工程属性,将用户态的虚拟内存从2GB扩大到3GB,用规避的方法加以解决,如何扩充,可以查看我之前写的专题文章:
如何配置32位C++程序启用大地址模式(将用户态虚拟内存从2GB扩充到3GB),以解决用户态虚拟内存不够用问题?(项目实战案例解析)
https://blog.csdn.net/chenlycly/article/details/1384605833)将程序升级到64位
可以将程序升级到64位,但这里有个问题。虽然大家使用的基本都是Win10及以上的系统,都是64位的,但有的用户的电脑用的系统比较老,比如还在用Win7,可能还是32位的。做成64位程序,则没法在32位系统中运行。
4)做成多进程模式可以做成多进程,但多进程加大了程序的开发难度,使得开发难度和成本都变高了。有的程序就做成了多进程模式,最具代表性的就是Chrome浏览器,就做成了多进程模式,每个进程负责执行不同的事务。
6、最后
本文从最近遇到的两个内存不足导致程序异常的案例出发,详细讲述了案例中涉及到的细节与要点,并对内存不足相关内容进行了展开和总结,有一定的实战参考价值,希望能帮到大家。
相关文章:

内存不足引发C++程序闪退崩溃问题的分析与总结
目录 1、内存不足一般出现在32位程序中 2、内存不足时会导致malloc或new申请内存失败 2.1、malloc申请内存失败,返回NULL 2.2、new申请内存失败,抛出异常 3、内存不足项目实战案例中相关细节与要点说明 3.1、内存不足导致malloc申请内存失败&#…...

C++ —— 以真我之名 如飞花般绚丽 - 智能指针
目录 1. RAII和智能指针的设计思路 2. C标准库智能指针的使用 2.1 auto_ptr 2.2 unique_ptr 2.3 简单模拟实现auto_ptr和unique_ptr的核心功能 2.4 shared_ptr 2.4.1 make_shared 2.5 weak_ptr 2.6 shared_ptr的缺陷:循环引用问题 3. shared_ptr 和 unique_…...

Linux中安装InfluxDB
什么是InfluxDB InfluxDB是一个开源的时间序列数据库,专为处理时间序列数据而设计。时间序列数据是指带有时间戳的数据点,例如传感器数据、应用程序日志、服务器指标等。InfluxDB 由 InfluxData 公司开发,广泛应用于物联网(IoT&am…...

nginx服务器实现上传文件功能_使用nginx-upload-module模块
目录 conf文件内容如下html文件内容如下上传文件功能展示 conf文件内容如下 #user nobody; worker_processes 1;error_log /usr/logs/error.log; #error_log /usr/logs/error.log notice; #error_log /usr/logs/error.log info;#pid /usr/logs/nginx.pid;events …...

ORB-SLAM2源码学习:Initializer.cc:Initializer::ComputeF21地图初始化——计算基础矩阵
前言 在平面场景我们通过求解单应矩阵H来求解位姿,但是我们在实际中常见的都是非平面场景, 此时需要用基础矩阵F求解位姿。 1.函数声明 cv::Mat Initializer::ComputeF21(const vector<cv::Point2f> &vP1, const vector<cv::Point2f>…...

C# 读取多条数据记录导出到 Word标签模板之图片输出改造
目录 应用需求 设计 范例运行环境 配置Office DCOM 实现代码 组件库引入 核心代码 调用示例 小结 应用需求 在我的文章《C# 读取多条数据记录导出到 Word 标签模板》里,讲述读取多条数据记录结合 WORD 标签模板输出文件的功能,原有输出图片的…...

NSSCTF web刷题
1 虽然找到了flag,但是我要怎么去改他的代码,让他直接输出flag呢? (好像是要得到他的json代码,这题不让看) 2 wllm应该就是他的密码,进入许可了 意思是服务器可以执行通过POST的请求方式传入参数为wllm的命令,那这就是典型的命令执行,当然,…...

对象排序得到方式
java实现 list 排序的方式,有三种 ① 对象实现Comparable 接口,然后代码里直接调用Collections.sort(list) ②使用内部类Comparator ③使用stream.sort 代码如下 实现Comparable接口的实体类 Data public class Student implements Comparable<Stud…...

Day2 洛谷1035+1047+1085+1089+1150+1151
零基础洛谷刷题记录 Day1 2024.11.18 Day2 2024.11.25 文章目录 零基础洛谷刷题记录1035:题目描述1035:解答代码1035:学习成果1047:题目描述(成功写出)1047:解答代码1047:学习成果1085…...

Linux:进程间通信之进程池和日志
一、进程池的设计 因为每一次我们要进行进程间通信都需要fork,和操作系统做交互是存在很大成本的,所以我们是不是可以提前fork出几个进程,然后当我们想要使用的时候直接去给他们安排任务,这样就减少了系统调用的次数从而提高了内存…...

详细介绍HTTP与RPC:为什么有了HTTP,还需要RPC?
目录 一、HTTP 二、RPC 介绍 工作原理 核心功能 如何服务寻址 如何进行序列化和反序列化 如何网络传输 基于 TCP 协议的 RPC 调用 基于 HTTP 协议的 RPC 调用 实现方式 优点和缺点 使用场景 常见框架 示例 三、问题 问题一:是先有HTTP还是先有RPC&…...

Paddle Inference部署推理(十二)
十二:Paddle Inference推理 (python)API详解 15. PredictorPool 类 PredictorPool 对 Predictor 进行了简单的封装,通过传入 config 和 thread 的数目来完成初始化,在每个线程中,根据自己的线程 id 直接从…...

外观模式 (Facade Pattern)
外观模式 (Facade Pattern) 外观模式是一种 结构型设计模式,通过为子系统中的一组接口提供一个统一的高层接口,简化了子系统的使用,让复杂系统更易于访问。 原理 核心思想: 提供一个 统一的接口 来访问子系统中的多个接口&#…...

人工智能-深度学习-Torch框架-手动构建回归流程
from sklearn.datasets import make_regression import math import random import torch from sklearn.datasets import make_regression: 导入make_regression函数,用于生成回归数据集。 import math: 导入math模块,用于进行数学计算,例如…...

SpringBoot源码解析(五):准备应用环境
SpringBoot源码系列文章 SpringBoot源码解析(一):SpringApplication构造方法 SpringBoot源码解析(二):引导上下文DefaultBootstrapContext SpringBoot源码解析(三):启动开始阶段 SpringBoot源码解析(四):解析应用参数args Sp…...

MySQL面试-1
InnoDB中ACID的实现 先说一下原子性是怎么实现的。 事务要么失败,要么成功,不能做一半。聪明的InnoDB,在干活儿之前,先将要做的事情记录到一个叫undo log的日志文件中,如果失败了或者主动rollback,就可以通…...

nginx配置不缓存资源
方法1 location / {index index.html index.htm;add_header Cache-Control no-cache,no-store;try_files $uri $uri/ /index.html;#include mime.types;if ($request_filename ~* .*\.(htm|html)$) {add_header Cache-Control "private, no-store, no-cache, must-revali…...

PHP导出EXCEL含合计行,设置单元格格式
PHP导出EXCEL含合计行,设置单元格格式,水平居中 垂直居中 public function exportSalary(Request $request){//水平居中 垂直居中$styleArray [alignment > [horizontal > Alignment::HORIZONTAL_CENTER,vertical > Alignment::VERTICAL_CE…...

RabbitMQ 之 死信队列
一、死信的概念 先从概念解释上搞清楚这个定义,死信,顾名思义就是无法被消费的消息,字面意思可以这样理 解,一般来说,producer 将消息投递到 broker 或者直接到 queue 里了,consumer 从 queue 取出消息进行…...

【创建型设计模式】单例模式
【创建型设计模式】单例模式 这篇博客接下来几篇都将阐述设计模式相关内容。 接下来的顺序大概是:单例模式、工厂方法模式、抽象工厂模式、建造者模式、原型模式。 一、什么是单例模式 单例模式是一种创建型设计模式,它保证一个类仅有一个实例&#…...

Charles抓包工具-笔记
摘要 概念: Charles是一款基于 HTTP 协议的代理服务器,通过成为电脑或者浏览器的代理,然后截取请求和请求结果来达到分析抓包的目的。 功能: Charles 是一个功能全面的抓包工具,适用于各种网络调试和优化场景。 它…...

Go语言使用 kafka-go 消费 Kafka 消息教程
Go语言使用 kafka-go 消费 Kafka 消息教程 在这篇教程中,我们将介绍如何使用 kafka-go 库来消费 Kafka 消息,并重点讲解 FetchMessage 和 ReadMessage 的区别,以及它们各自适用的场景。通过这篇教程,你将了解如何有效地使用 kafk…...

【论文笔记】Number it: Temporal Grounding Videos like Flipping Manga
🍎个人主页:小嗷犬的个人主页 🍊个人网站:小嗷犬的技术小站 🥭个人信条:为天地立心,为生民立命,为往圣继绝学,为万世开太平。 基本信息 标题: Number it: Temporal Grou…...

C语言菜鸟入门·关键字·int的用法
目录 1. int关键字 1.1 取值范围 1.2 符号类型 1.3 运算 1.3.1 加法运算() 1.3.2 减法运算(-) 1.3.3 乘法运算(*) 1.3.4 除法运算(/) 1.3.5 取余运算(%) 1.3.6 自增()与自减(--) 1.3.7 位运算 2. 更多关键字 1. int关键字 int 是一个关键字࿰…...

基于企业微信客户端设计一个文件下载与预览系统
在企业内部沟通与协作中,文件分享和管理是不可或缺的一部分。企业微信(WeCom)作为一款广泛应用于企业的沟通工具,提供了丰富的API接口和功能,帮助企业进行高效的团队协作。然而,随着文件交换和协作的日益增…...

昇思MindSpore第七课---文本解码原理
1. 文本解码原理 文本解码是将模型的输出(通常是概率分布或词汇索引)转换为可读的自然语言文本的过程。在生成文本时,常见的解码方法包括贪心解码、束搜索(BeamSearch)、随机采样等。 2 实践 2.1 配置环境 安装mindn…...

C# 数据结构之【图】C#图
1. 图的概念 图是一种重要的数据结构,用于表示节点(顶点)之间的关系。图由一组顶点和连接这些顶点的边组成。图可以是有向的(边有方向)或无向的(边没有方向),可以是加权的ÿ…...

传输控制协议(TCP)和用户数据报协议(UDP)
一、传输控制协议(TCP) 传输控制协议(Transmission Control Protocol,TCP)是一种面向连接的、可靠的、基于字节流的传输层通信协议,由 IETF 的 RFC 793 定义。 它通过三次握手建立连接,确保数…...

【Python爬虫】Scrapy框架实战---百度首页热榜新闻
如何利用Scrapy框架实战提取百度首页热榜新闻的排名、标题和链接 一、安装Scrapy库 二、创建项目(以BaiduSpider为例) scrapy startproject BaiduSpider生成每个文件的功能: 二、 创建爬虫脚本(爬虫名:newsÿ…...

采用python3.12 +django5.1 结合 RabbitMQ 和发送邮件功能,实现一个简单的告警系统 前后端分离 vue-element
一、开发环境搭建和配置 #mac环境 brew install python3.12 python3.12 --version python3.12 -m pip install --upgrade pip python3.12 -m pip install Django5.1 python3.12 -m django --version #用于检索系统信息和进程管理 python3.12 -m pip install psutil #集成 pika…...