C++ 内存分配(new,operator new)详解

Posted 隐无影

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C++ 内存分配(new,operator new)详解相关的知识,希望对你有一定的参考价值。

出处:http://blog.csdn.net/wudaijun/article/details/9273339


    本文主要讲述C++ new运算符和operator new, placement new之间的种种关联,new的底层实现,以及operator new的重载和一些在内存池,STL中的应用。

一 new运算符和operator new():

     new:指我们在C++里通常用到的运算符,比如A* a = new A;  对于new来说,有new和::new之分,前者位于std      operator new():指对new的重载形式,它是一个函数,并不是运算符。对于operator new来说,分为全局重载和类重载,全局重载是void* ::operator new(size_t size),在类中重载形式 void* A::operator new(size_t size)。还要注意的是这里的operator new()完成的操作一般只是分配内存,事实上系统默认的全局::operator new(size_t size)也只是调用malloc分配内存,并且返回一个void*指针。而构造函数的调用(如果需要)是在new运算符中完成的
     先简单解释一下new和operator new之间的关系:      关于这两者的关系,我找到一段比较经典的描述(来自于www.cplusplus.com 见参考文献): operator new can be called explicitly as a regular function, but in C++, new is an operator with a very specific behavior: An expression with the new operator, first calls function operator new (i.e., this function) with the size of its type specifier as first argument, and if this is successful, it then automatically initializes or constructs the object (if needed). Finally, the expression evaluates as a pointer to the appropriate type.      比如我们写如下代码:      A* a = new A;      我们知道这里分为两步:1.分配内存,2.调用A()构造对象。事实上,分配内存这一操作就是由operator new(size_t)来完成的,如果类A重载了operator new,那么将调用A::operator new(size_t ),如果没有重载,就调用::operator new(size_t ),全局new操作符由C++默认提供。因此前面的两步也就是:1.调用operator new 2.调用构造函数。这里再一次提出来是因为后面关于这两步会有一些变形,在关于placement new那里会讲到。先举个简单例子
[cpp]  view plain copy print ?
  1. //平台:Visual Stdio 2008  
  2. #include<iostream>  
  3. class A  
  4.   
  5. public:  
  6.      A()  
  7.        
  8.           std::cout<<"call A constructor"<<std::endl;  
  9.        
  10.   
  11.      ~A()  
  12.        
  13.           std::cout<<"call A destructor"<<std::endl;  
  14.        
  15.   
  16. int _tmain(int argc, _TCHAR* argv[])  
  17.   
  18.   
  19.      A* a = new A;  
  20.      delete a;  
  21.   
  22.      system("pause");  
  23.      return 0;  
  24.   



下面我们跟踪一下A反汇编代码,由于Debug版本反汇编跳转太多,因此此处通过Release版本在A* a = new A;处设断点反汇编: 在Release版本中,构造函数和析构函数都是直接展开的。
[cpp]  view plain copy print ?
  1.     A* a = new A;  
  2. 01301022  push        1    ;不含数据成员的类占用一字节空间,此处压入sizeof(A)  
  3. 01301024  call        operator new (13013C2h) ;调用operator new(size_t size)  
  4. 01301029  mov         esi,eax ;返回值保存到esi  
  5. 0130102B  add         esp,4 ;平衡栈  
  6. 0130102E  mov         dword ptr [esp+8],esi ;  
  7. 01301032  mov         dword ptr [esp+14h],0   
  8. 0130103A  test        esi,esi ;在operator new之后,检查其返回值,如果为空(分配失败),则不调用A()构造函数  
  9. 0130103C  je          wmain+62h (1301062h) ;为空 跳过构造函数部分  
  10. 0130103E  mov         eax,dword ptr [__imp_std::endl (1302038h)] ;构造函数内部,输出字符串  
  11. 01301043  mov         ecx,dword ptr [__imp_std::cout (1302050h)]   
  12. 01301049  push        eax    
  13. 0130104A  push        offset string "call A constructor" (1302134h)   
  14. 0130104F  push        ecx    
  15. 01301050  call        std::operator<<<std::char_traits<char> > (13011F0h)   
  16. 01301055  add         esp,8   
  17. 01301058  mov         ecx,eax   
  18. 0130105A  call        dword ptr [__imp_std::basic_ostream<char,std::char_traits<char> >::operator<< (1302040h)]   
  19. 01301060  jmp         wmain+64h (1301064h) ;构造完成,跳过下一句  
  20. 01301062  xor         esi,esi ;将esi置空,这里的esi即为new A的返回值  
  21. 01301064  mov         dword ptr [esp+14h],0FFFFFFFFh   
  22.     delete a;  
  23. 0130106C  test        esi,esi ;检查a是否为空  
  24. 0130106E  je          wmain+9Bh (130109Bh) ;如果为空,跳过析构函数和operator delete  
  25. 01301070  mov         edx,dword ptr [__imp_std::endl (1302038h)] ;析构函数 输出字符串  
  26. 01301076  mov         eax,dword ptr [__imp_std::cout (1302050h)]   
  27. 0130107B  push        edx    
  28. 0130107C  push        offset string "call A destructor" (1302148h)   
  29. 01301081  push        eax    
  30. 01301082  call        std::operator<<<std::char_traits<char> > (13011F0h)   
  31. 01301087  add         esp,8   
  32. 0130108A  mov         ecx,eax   
  33. 0130108C  call        dword ptr [__imp_std::basic_ostream<char,std::char_traits<char> >::operator<< (1302040h)]   
  34. 01301092  push        esi  ;压入a   
  35. 01301093  call        operator delete (13013BCh) ;调用operator delete   
  36. 01301098  add         esp,4   
  37. 通过反汇编可以看出A* = new A包含了operator new(sizeof(A))和A()两个步骤(当然,最后还要将值返回到a)  
  38.          delete a包含了~A()和operator delete(a)两个步骤。  



二 operator new的三种形式:

operator new有三种形式:
throwing (1)
void* operator new (std::size_t size) throw (std::bad_alloc);
nothrow (2)
void* operator new (std::size_t size, const std::nothrow_t& nothrow_value) throw();
placement (3)
void* operator new (std::size_t size, void* ptr) throw();
(1)(2)的区别仅是是否抛出异常,当分配失败时,前者会抛出bad_alloc异常,后者返回null,不会抛出异常。它们都分配一个固定大小的连续内存。 用法示例: A* a = new A; //调用throwing(1) A* a = new(std::nothrow) A; //调用nothrow(2) (3)是placement new,它也是对operator new的一个重载,定义于<new>中,它多接收一个ptr参数,但它只是简单地返回ptr。其在new.h下的源代码如下:
[cpp]  view plain copy print ?
  1. #ifndef __PLACEMENT_NEW_INLINE  
  2. #define __PLACEMENT_NEW_INLINE  
  3. inline void *__cdecl operator new(size_tvoid *_P)  
  4.         return (_P);   
  5. #if     _MSC_VER >= 1200  
  6. inline void __cdecl operator delete(void *, void *)  
  7.     return;   
  8. #endif  
  9. #endif  


那么它究竟有什么用呢?事实上,它可以实现在ptr所指地址上构建一个对象(通过调用其构造函数),这在内存池技术上有广泛应用。 它的调用形式为: new(p) A(); //也可用A(5)等有参构造函数。 前面说到,new运算符都会调用operator new,而这里的operator new(size_t, void*)并没有什么作用,真正起作用的是new运算符的第二个步骤:在p处调用A构造函数。这里的p可以是动态分配的内存,也可以是栈中缓冲,如char buf[100]; new(buf) A();
我们仍然可以通过一个例子来验证: [cpp]  view plain copy print ?
  1. #include <iostream>  
  2. class A  
  3.   
  4. public:  
  5.     A()  
  6.       
  7.         std::cout<<"call A constructor"<<std::endl;  
  8.       
  9.   
  10.     ~A()  
  11.       
  12.         std::cout<<"call A destructor"<<std::endl;  
  13.       
  14. ;  
  15. C++ 内存分配(new,operator new)详解

    C++ 内存分配(new,operator new)详解

    c++内存分配(new,operator new)详解

    C++ 内存分配(new,operator new)详解

    C++ 内存分配(new,operator new)详解

    operator new 如何知道内存已分配[重复]