C++中new申请内存失败抛出异常的处理

Posted dvlinker

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++中new申请内存失败抛出异常的处理相关的知识,希望对你有一定的参考价值。

       正常情况下,执行new操作时都能申请到内存,并返回一个有效的地址。但在某些异常的情况下,会出现申请内存失败的情况。

       new操作返回NULL申请不到内存,或者new时抛出异常,可能有以下几种原因引起的
1)用户态的内存已经达到了上限,申请不到内存了
有可能是虚拟内存占用太多,也有可能代码中有内存泄露,导致用户态的内存被消耗完了。
2)进程中的内存碎片过多
如果进程中在大量的new和delete,产生了大量的小块内存碎片,可用的内存大多是一小块一小块的小内存块,而要申请的是一块长度很长的内存,因为到处是内存碎片,没有这么一大块连续的可用内存,可能就会导致内存申请失败的。
3)发生堆内存越界,导致堆内存被破坏,导致new或delete产生异常(此时new不会返回NULL,会抛出异常)
系统在分配堆内存时,会在给用户的堆内存块的头部和尾部添加一些头尾信息的,这些头尾信息是用来管理这些堆内存的,如果这些头尾信息被越界篡改,会导致后续的new或delete产生异常,导致程序崩溃。

       关于new抛出异常的说明,可以参加这篇文章:
Decoding the parameters of a thrown C++ exception (0xE06D7363)

      对于new抛出异常的场景,可以使用下面两种方法来处理:

1)使用try-catch来捕获:

try

    int *p = new int[10000000000];

catch (bad_alloc &e)

	cout << e.what() << endl;

 2)强制new不抛出异常,此时会返回空指针

int *p = new (std::nothrow) int[1000000000];
if (p == NULL)

	cout << "bad alloc" << endl;

      其实很多时候出现new内存失败的问题后,会导致业务不能正常进行,即使通过上述方法能让程序不出现崩溃,但程序继续运行的意义已经不大了。 所以很多开源软件在new返回空或者抛出异常时,会直接调用abort接口将当前进程强制结束掉,给用户的感觉就是程序发生异常闪退,其实是程序主动强制结束进程的。此时通过windbg挂载到目标进程上可以感知到,可以通过函数调用堆栈看一下时何种原因导致的。具体是何种原因导致的,就要参考上面所说的三种原因了!

以上是关于C++中new申请内存失败抛出异常的处理的主要内容,如果未能解决你的问题,请参考以下文章

set_new_handle

两例典型的C++软件异常排查实例分享

malloc 和 new 的区别

3.2 模拟面试

动态内存分配

八定制new和delete