C++ - 定义自定义新建和删除运算符时make_shared
C++ - make_shared when custom new and delete operators are defined
关于Scot Meyers的"Effective Modern C++"一书,以及第21项:"更喜欢std::make_unique和std::make_shared直接使用new":
"一些类定义了自己的运算符new
和运算符delete
版本。通常,特定于类的例程仅用于分配和释放与类对象大小精确匹配的内存块。这样的例程不适合std::shared_ptr
对自定义分配(通过std::allocate_shared
)和释放(通过自定义删除器)的支持,因为std::allocate_shared
请求的内存量不是动态分配对象的大小,而是该对象的大小加上控制块的大小。因此,使用 make 函数创建具有特定于类的运算符new
和运算符delete
版本的类型对象通常是一个糟糕的主意。
如果自定义new
和delete
在与标准new
和delete
相同的位置调用,为什么这对allocate_shared
/make_shared
来说是一个问题?
建设: 运算符new
仅用于构造资源对象,但make_shared
/allocate_shared
构造 constrol 块。
破坏: 无论是否指定了自定义删除程序函数,调用delete
时,应仅删除资源对象。Cntrol阻断取决于参考和弱计数。
那么为什么会有这样的句子:"这样的例程不适合std::shared_ptr
对自定义分配(通过std::allocate_shared
)和释放(通过自定义删除器)的支持,因为std::allocate_shared
请求的内存量不是动态分配对象的大小,而是该对象的大小加上控制块的大小。
std::make_unique
很好,因为它不需要任何额外的数据(在一般情况下)。
问题std::make_shared
。为了使std::shared_ptr
工作,需要一些额外的数据:强引用计数器、弱引用计数器、删除器函数指针。
这意味着当你这样做时:std::shared_ptr<Foo>{raw_pointer}
,std::shared_ptr
将分配自己的缓冲区来保存这些数据。 现在分配非常昂贵,因此引入了std::make_shared
以优化堆的使用。它分配内存,该内存将在单个内存块中并排保存shared_ptr
数据和指向的项目。它就地构造指向对象。
因此,当您对某个类具有自定义运算符new
时,std::make_shared
不会使用它。另一方面,如果您将原始指针传递给std::shared_ptr
那么您将有两个内存块。
- 为什么比较运算符如此快速
- C++映射:具有自定义类的运算符[]不起作用(总是返回0)
- 使用C++中的模板和运算符重载执行矩阵运算
- 为什么这个运算符<重载函数对 STL 算法不可见?
- 增量运算符与后缀混淆
- 一个关于在C++中重载布尔运算符的问题
- 运算符C++ "delete []"仅删除 2 个前值
- 模板类无法识别友元运算符
- 我可以使用条件运算符初始化C风格的字符串文字吗
- 关闭||运算符优化
- 通过继承类使用来自不同命名空间的运算符
- C++Cast运算符过载
- 如何使用AngelScript注册SFML Vector2运算符
- 重载元组索引运算符-C++
- 如何使用重载的相等(==)运算符向测试用例添加描述
- 为什么Mat类的两个对象可以在不重载运算符+的情况下添加
- 多个If语句与使用逻辑运算符计算条件的单个语句的比较
- 布尔比较运算符是如何在C++中工作的
- 重载运算符new[]的行为取决于析构函数
- 如何防止clang格式在流运算符调用之间添加换行符<<