cpp中容器push_back和emplace_back的区别
1. 主要区别
当使用push_back时会先调用类的有参构造函数创建一个临时变量,再将这个元素拷贝或者移动到容器之中,而emplace_back则是直接在容器尾部进行构造比push_back少进行一次构造函数调用。在大部分场景中emplace_back可以替换push_back,但是push_back会比emplace_back更加安全,emplace_back只能用于直接在容器中构造新元素的情况,如果要将现有的对象添加到容器中则需要使用push_back
简单来说就是
-
push_back:会先构造一个临时对象(调用有参构造函数),然后将其复制或移动到容器中。 -
emplace_back:会直接在容器尾部调用构造函数构造对象,避免临时对象的创建和拷贝或移动。
2. 示例说明
#include <iostream>
#include <vector>
#include <string>
class Person {
public:
std::string name;
int age;
// 有参构造函数
Person(const std::string& name, int age) : name(name), age(age) {
std::cout << "Constructed Person: " << name << ", age " << age << std::endl;
}
// 拷贝构造函数
Person(const Person& other) {
name = other.name;
age = other.age;
std::cout << "Copy constructed Person: " << name << ", age " << age << std::endl;
}
// 移动构造函数
Person(Person&& other) noexcept : name(std::move(other.name)), age(other.age) {
std::cout << "Move constructed Person: " << name << ", age " << age << std::endl;
}
};
int main() {
std::vector<Person> people;
std::cout << "--- Using push_back ---" << std::endl;
Person p1("Alice", 30); // 创建一个Person对象p1
people.push_back(p1); // 使用push_back传递现有的对象,需要进行拷贝构造
std::cout << "--- Using emplace_back ---" << std::endl;
people.emplace_back("Bob", 25); // 使用emplace_back直接在容器中构造对象,不会创建临时对象
return 0;
}
输出的结果是
--- Using push_back ---
Constructed Person: Alice, age 30 // 构造 p1
Copy constructed Person: Alice, age 30 // 将 p1 拷贝到 vector 中
--- Using emplace_back ---
Constructed Person: Bob, age 25 // 直接在 vector 中构造 Bob 对象
关于push_back:
首先创建了一个 Person 对象 p1(调用有参构造函数)。然后在调用 push_back(p1) 时,p1 被拷贝到 vector 中,这触发了拷贝构造函数,导致了额外的对象拷贝。
关于emplace_back:
emplace_back("Bob", 25) 直接在 vector 中创建 Person("Bob", 25),没有创建临时对象,也不需要调用拷贝或移动构造函数。只会调用一次有参构造函数,性能上更优。
总结来说,
push_back 更安全:如果你有现有的对象,像 p1,你应该使用 push_back,因为它可以将现有的对象添加到容器中(拷贝或移动)。emplace_back 不能处理现有对象。
emplace_back 更高效:emplace_back 用于直接在容器中构造对象,避免额外的拷贝或移动构造。

被折叠的 条评论
为什么被折叠?



