set_new_handler

1. 问题

动态申请数组可以无限大吗?如果动态分配一个很大的数组会出现什么情况?

int main(){
  int *p = new int[0x1fffffff];
}

执行结果:程序终止

terminate called after throwing an instance of 'std::bad_alloc'
  what():  std::bad_alloc

我们可以尝试捕获异常

#include <iostream>
int main(){
  try{
    int *p = new int[0x1fffffff];
  }catch(std::exception& e){
    std::cout << e.what() << std::endl;
  }
}

执行结果:程序退出

std::bad_alloc

2. 内存申请失败

当申请内存不够时就会抛出out of memory的异常。但是有时,我们希望能够调用自己定制的异常处理函数,优先处理。这就是set_new_handler

// 定义在<new>中
namespace std{
    typedef void (*new_handler)();
    std::new_handler set_new_handler( std::new_handler new_p );
}

例子

#include <iostream>
void out_of_memory() {  
    std::cout<<"out of memory!"<<std::endl;  
}  
  
int main() {  
    std::set_new_handler(out_of_memory);  
    try{
      int *p = new int[0x1fffffff];
    }catch(std::exception& e){
      std::cout << e.what() << std::endl;
    }    
    return 0;  
}

执行结果:

out of memory!
out of memory!
out of memory!
out of memory!
out of memory!
...

分析

operator new申请一个内存失败的时候,它会进行如下的处理步骤:

  1. 如果存在客户指定的处理函数,则调用处理函数(new_handler),如果不存在则抛出一个异常。
  2. 继续申请内存分配请求。
  3. 判断申请内存是否成功,如果成功则返回内存指针,如果失败转向处理步骤1

operator new无法满足内存分配需求时,它会不断调用new_handler函数,直到找到足够的内存。

处理

如何避免重复调用new_handler函数?

#include <iostream>
void out_of_memory() {  
    std::cout<<"out of memory!"<<std::endl;  
    std::set_new_handler(NULL);  
}  
  
int main() {  
    std::set_new_handler(out_of_memory);  
    try{
      int *p = new int[0x1fffffff];
    }catch(std::exception& e){
      std::cout << e.what() << std::endl;
    }     
    return 0;  
}

执行结果:

out of memory!
std::bad_alloc

3. set_new_handler机制的运行原理

operator new无法满足某一个内存分配的需求的时候,它会抛出一个异常。在这之前,它会先调用一个用户制定的错误处理函数new_handler。为了指定这个处理内存不足的函数,用户需要调用标准库程序函数set_new_handler

4. 设计new_handler函数

  1. 删除其它无用的内存
    使系统具有可以更多的内存可以使用,为下一步的内存申请作准备。
    实现此策略的办法是:程序一开始执行就分配一大块内存,当new_handler被调用时,将它们释放还给程序使用。
  2. 设置另外一个new_handler
    如果当前的new_handler不能够做到更多的内存申请操作,或者它知道另外一个new_handler可以做到,则可以调用set_new_handler函数设置另外一个new_handler,这样在operator new下一次调用的时候,可以使用这个新的new_handler
  3. 卸载new_handler,使operator new在下一次调用的时候,因为new_handler为空抛出内存申请异常。
  4. new_handler抛出自定义的异常。
  5. 不再返回,调用abort()或者exit()退出程序。

类的new_handler

上面的new_handler是全局的new_handler,对类也可以使用。

#include <iostream>
void out_of_memory() {  
    std::cout<<"out of memory!"<<std::endl;  
    std::set_new_handler(NULL);  
}  
class Test{};
int main() {  
    std::set_new_handler(out_of_memory);  
    try{
      Test* p = new Test[0x1fffffff];
    }catch(std::exception& e){
      std::cout << e.what() << std::endl;
    }     
    return 0;  
}

但是如果要实现类自己的new_handler,就需要添加如下处理。

1. 添加静态的new_handler函数和operator new函数

class Test{
public:
  static std::new_handler set_new_handler(std::new_handler p) throw();
  static void* operator new(std::size_t size) throw(std::bad_alloc);
private:
  static std::new_handler currentHandler;
  int data[0x1fffffff];
};
new_handler Test::currentHandler = 0;  

2. 实现set_new_handler函数

new_handler Test::set_new_handler(new_handler p) throw() {  
    new_handler oldHandler = currentHandler;  
    currentHandler = p;  
    return oldHandler;  
}

3. 实现operator new函数

void* Test::operator new(std::size_t size) throw(std::bad_alloc) {
  HandlerHolder holder(std::set_new_handler(currentHandler));
  return ::operator new(size);
}

其中,HandlerHolder是一个简单的资源处理类。

class HandlerHolder{
public:
    explicit HandlerHolder(std::new_handler nh):handler(nh){}
    ~HandlerHolder(){
        std::set_new_handler(handler);
    }
private:
    std::new_handler handler;
    HandlerHolder(HandlerHolder const&);
    HandlerHolder& operator=(HandlerHolder const&);
};

测试代码

#include <iostream>
void out_of_memory() {  
    std::cerr<<"out of memory!"<<std::endl;  
    std::set_new_handler(NULL);  
}  
void class_out_of_memory() {  
    std::cerr<<"class out of memory!"<<std::endl;  
    std::set_new_handler(NULL);  
} 
int main(){
    std::set_new_handler(out_of_memory);
    Test::set_new_handler(class_out_of_memory);
    try{
      Test* p = new Test;
    }catch(std::exception& e){
      std::cout << e.what() << std::endl;
    }     
    return 0;   
}

执行结果

class out of memory!

如果把Test* p = new Test;改为int* p = int[0x1fffffff],执行结果是

out of memory!

问题
如果是Test* p = new Test[0x1fffffff];呢?

扩展阅读

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,529评论 5 475
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,015评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,409评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,385评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,387评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,466评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,880评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,528评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,727评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,528评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,602评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,302评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,873评论 3 306
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,890评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,132评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,777评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,310评论 2 342

推荐阅读更多精彩内容