百科问答小站 logo
百科问答小站 font logo



C++ 中 new 和定位 new 的返回值都是 void*,却为什么可以被赋值给不同类型的指针? 第1页

  

user avatar   breakerzhao 网友的相关建议: 
      

这里有两个不同的东西,new expression(也叫 new operator)和 operator new()。

new expression 是语言内建机制,不能重载,我们常写的 new T、new T(param) 等就是这个,它返回有类型的指针 T*。new T 做两件固定的事,第一步调用 operator new(),第二步调用 T 的构造函数,这两步都由编译器自动产生。

operator new() 可以重载,它返回 void*,是一块未初始化的原始内存,随后控制返回到内建的 new expression 中。

placement new 的 operator new() 默认实现如下。它的第一个 size_t 是个必须有,但虚置无用的参数。它的功能就是返回已由用户提供的原始内存地址。

       void* operator new(size_t, void* loc) {     return loc; }      

常规的 new expression 执行过程如下:

       #include <new>  class Widget { public:     Widget(Param p) { /*...*/ }     ~Widget() { /*...*/ }     //... };  // 常规的 new expression Widget* pw = new Widget(param);  // 大致等价于以下过程: // [1] 直接调用 operator new() 函数分配原始内存 void* raw = operator new(sizeof(Widget)); // 上面作用类似下面这句 void* raw = malloc(sizeof(Widget));  // [2] 利用 placement new 调用 Widget 的构造函数 // 这里也是本题的关键: new expression 会将 void* 转换为有类型的指针 Widget* Widget* pw = new(raw) Widget(param);      

相应地,常规的 delete expression 执行过程如下:

       // 常规的 delete expression delete pw;  // 大致等价于以下过程: // [1] 手工调用 Widget 的析构函数 pw->~Widget(); // [2] 直接调用 operator delete() 函数释放原始内存 operator delete(raw); // 上面作用类似下面这句 free(raw);      

以上模拟的是 Widget 构造函数中没有抛出异常时的 new-delete 过程。

没有 placement delete expression 语法,只有 placement operator delete() 函数。在用 placement new 时,只要考虑手工调用 Widget 的析构函数即可,而 placement operator delete() 是 Widget 的构造函数中抛出异常时,由编译器自动调用的。




  

相关话题

  C++中,自定义函数写在main之前更好,还是之后更好? 
  关于Qt性能的损失,有没有一个可以量化的概念? 
  c++的单例模式为什么不直接全部使用static,而是非要实例化一个对象? 
  指针数组初始化为 nullptr 和直接使用 memcpy 有什么区别? 
  C++可否将父类的对象cast子类,并调用子类的private函数? 
  C++异常处理写的代码太丑怎么办? 
  C++函数收到一个指针T* ptr,没有其他信息,如何判断应该用delete还是delete[]? 
  C++ 和 Java 同样是静态语言,为什么 Java 的代码提示可以做的十分强大? 
  在指针的概念中为什么有几个描述是这样的呢? 
  C/C++中,设计的时候字符/字符串为什么要加引号? 

前一个讨论
C++整型有__int8、__int16、__int32等等,为什么还要short、int、long?
下一个讨论
游戏《原神》 北斗邀约中「真正的交易」是指的什么?





© 2025-01-31 - tinynew.org. All Rights Reserved.
© 2025-01-31 - tinynew.org. 保留所有权利