Do we really need placement new-expressions?(我们真的需要放置新的表达方式吗?)
问题描述
我正在尝试理解C++中的placement new-expressions。
This Stack Overflow answer说明T* p = new T(arg);
等同于
void* place = operator new(sizeof(T)); // storage allocation
T* p = new(place) T(arg); // object construction
,delete p;
等同于
p->~T(); // object destruction
operator delete(p); // storage deallocation
为什么我们需要T* p = new(place) T(arg);
中的新表达式来构造对象,下面的不是等价的吗?
T* p = (T*) place;
*p = T(arg);
推荐答案
首先要注意的是*p = T(arg);
是赋值,而不是构造。
现在我们来读一下标准([basic.life]/1):
.
T
类型的对象的生存期从以下时间开始:
- 获得了类型
T
的正确对齐和大小的存储,并且- 其初始化(如果有)完成(包括空初始化)
对于一般类型T
,如果使用放置new
,则可以完成初始化,但情况并非如此。正在做
void* place = operator new(sizeof(T));
T* p = (T*)place;
不开始*p
的生存期。
同一节内容为([basic.life]/6):
.在对象的生命周期开始之前,但在分配了该对象将占用的存储空间之后...表示对象将位于的存储位置的地址的任何指针...Locate可以使用,但仅限于有限的方式。..。在以下情况下,程序具有未定义的行为: ...
- 指针用于访问非静态数据成员或调用对象的非静态成员函数, ...
operator=
是非静态成员函数,执行*p = T(arg);
等效于p->operator=(T(arg))
会导致未定义的行为。
new
,则不会调用构造函数,也不会初始化该指针(complete example)。
这篇关于我们真的需要放置新的表达方式吗?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:我们真的需要放置新的表达方式吗?
基础教程推荐
- C++,'if' 表达式中的变量声明 2021-01-01
- 运算符重载的基本规则和习语是什么? 2022-10-31
- 您如何将 CreateThread 用于属于类成员的函数? 2021-01-01
- 设计字符串本地化的最佳方法 2022-01-01
- 调用std::Package_TASK::Get_Future()时可能出现争用情况 2022-12-17
- 如何定义双括号/双迭代器运算符,类似于向量的向量? 2022-01-01
- C++ 程序在执行 std::string 分配时总是崩溃 2022-01-01
- C++ 标准:取消引用 NULL 指针以获取引用? 2021-01-01
- 什么是T&&(双与号)在 C++11 中是什么意思? 2022-11-04
- 如何在 C++ 中处理或避免堆栈溢出 2022-01-01