C++ std::vector 的 emplace_back 能否完全取代 push_back

本文详细对比了STL中vector容器成员函数emplace_back与push_back的区别,通过实例讲解了两者在性能上的差异,以及在特殊场景下emplace_back可能存在的风险。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

区别:

push_back:先在调用处构造一次 class,传递进 push_back 内后再进行拷贝到缓冲区。

emplace_back:在内部直接将构造 class 的参数转发构造到缓冲区。

 

如果以上说法不好理解,那么用代码来表示。

// 该 Class 支持隐式构造
class Class
{
public:
	Class(int a) : _a(a) {}

	int _a;
};

vector<Class> v;

// A
v.push_back(10);
// B
v.push_back(Class(10));

// C
v.emplace_back(10);
// D
v.emplace_back(Class(10));

其中:

A 和 B 是等效的。

C 和 D 不等效,且 C 比 D 快。

 

可见 push_back 比 emplace_back 多了一次拷贝操作。

emplace_back 的效率比 push_back 高。

 

那么问题来了,emplace_back 能否完全取代 push_back?

答案是:在大部分情况下是可以的。

 

那么小部分情况指的是什么呢?

来看例子

// 假定下面使用的 unique_ptr 支持隐式构造

vector<unique_ptr<int>> v;

v.push_back(new int(10));
v.emplace_back(new int(10));

如果对该 vector 的本次操作需要 resize,且 resize 抛出了一个异常。

在这种情况下,push_back 是绝对安全的,而 emplace_back 则有可能发生内存泄露。

因为使用 push_back,在抛出异常前,unique_ptr 已经构造成功,可以正常释放内存。

但是如果使用 emplace_back,在抛出异常前,uniquer_ptr 还未进行构造,这里就只是单纯的一个指针,不会被自动释放。

 

参考资料:

https://blog.csdn.net/xiaolewennofollow/article/details/52559364

https://stackoverflow.com/questions/22080290/are-there-any-cases-where-it-is-incorrect-to-replace-push-back-with-emplace-back

 

如文章内有任何不正之处,欢迎在评论区指出。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值