- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我想知道 std::vector
的 emplace_back
和 push_back
方法在使用原始标量类型时是否有任何不同,例如std::uint32_t
或 std::uint8_t
。凭直觉,我猜想在编译之后,这两种变体会在此处产生相同的字节码:
void copyListContent(std::uint8_t * list, std::size_t nElems,
std::vector<std::uint8_t> & vec)
{
vec.clear();
vec.reserve(nElems);
for (std::size_t i = 0; i < nElems; ++i)
{
//variant 1:
vec.push_back(list[i]);
//variant 2:
vec.emplace_back(list[i]);
}
}
如果那应该已经错了,请纠正我...
现在,当我问自己如果“列表”和 vector 的类型不匹配会发生什么时,我开始纠结:
void copyListContent(std::uint8_t * list, std::size_t nElems,
std::vector<std::uint32_t> & vec)
{
//... same code as above
}
std::uint8_t
元素在将它们放入 vector 时将被转换为 std::uint32_t
(使用 emplace_back
或 push_back
),所以我想知道这是否会触发调用某些“构造函数”?在那种情况下,emplace_back
会不会更有效率,因为它会避免构造一个将被复制的临时对象?还是这些隐式转换没有任何区别,emplace_back
和 push_back
的行为相同?
所以,我问自己,还有你:对于像这样的原始类型,emplace_back
和 push_back
是否总是表现相似?
作为一个模糊的猜测,我会说“可能是”,但我对 C++ 内部知识还不够了解,无法为自己可靠地回答这个问题。我很乐意了解这种情况下的工作原理 - 非常感谢!
最佳答案
GCC 将两个版本的代码编译成相同的结果程序集 (Godbolt.org):
#include<vector>
void push(std::vector<int> & vec, int val) {
vec.push_back(val);
}
对比
#include<vector>
void push(std::vector<int> & vec, int val) {
vec.emplace_back(val);
}
两者都会产生以下程序集:
push(std::vector<int, std::allocator<int> >&, int):
push r15
push r14
push r13
push r12
push rbp
push rbx
sub rsp, 24
mov rbx, QWORD PTR [rdi+8]
cmp rbx, QWORD PTR [rdi+16]
je .L2
mov DWORD PTR [rbx], esi
add rbx, 4
mov QWORD PTR [rdi+8], rbx
add rsp, 24
pop rbx
pop rbp
pop r12
pop r13
pop r14
pop r15
ret
.L2:
mov r12, QWORD PTR [rdi]
mov r14, rbx
mov ecx, esi
mov rbp, rdi
sub r14, r12
mov rax, r14
sar rax, 2
je .L9
lea rdx, [rax+rax]
mov r15, -4
cmp rax, rdx
ja .L4
movabs rsi, 4611686018427387903
cmp rdx, rsi
jbe .L19
.L4:
mov rdi, r15
mov DWORD PTR [rsp], ecx
call operator new(unsigned long)
mov ecx, DWORD PTR [rsp]
mov r13, rax
add r15, rax
.L5:
lea rax, [r13+4+r14]
mov DWORD PTR [r13+0+r14], ecx
mov QWORD PTR [rsp], rax
cmp rbx, r12
je .L6
mov rdx, r14
mov rsi, r12
mov rdi, r13
call memmove
.L7:
mov rdi, r12
call operator delete(void*)
.L8:
mov QWORD PTR [rsp+8], r13
movq xmm0, QWORD PTR [rsp+8]
mov QWORD PTR [rbp+16], r15
movhps xmm0, QWORD PTR [rsp]
movups XMMWORD PTR [rbp+0], xmm0
add rsp, 24
pop rbx
pop rbp
pop r12
pop r13
pop r14
pop r15
ret
.L6:
test r12, r12
je .L8
jmp .L7
.L9:
mov r15d, 4
jmp .L4
.L19:
xor r15d, r15d
xor r13d, r13d
test rdx, rdx
je .L5
lea r15, [0+rax*8]
jmp .L4
正如您可能已经推断的那样,在处理具有更复杂的构造/复制/移动行为的类型时,这不是您可以依赖的行为,但对于原始类型,差异可以忽略不计。
话虽如此,但有一种情况可能会有所不同:
std::vector<int16_t> vec;
size_t seed = 0x123456789abcdef;
vec.push_back(seed);
对比
vec.emplace_back(seed);
在(适当优化的)编译器中,这两个汇编代码可能是相同的,但是您会从编译器得到不同的缩小警告(或错误,如果您强制警告导致编译失败)。后者更有可能给出难以诊断的警告消息,因为错误将源自 <vector>
内部,而不是调用的任何 .cpp 文件内部。
关于c++ - 原始类型的 emplace_back 与 push_back,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48158286/
这个问题在这里已经有了答案: Passing by value vs const & and && overloads (3 个答案) 关闭 8 年前。 为什么push_back的函数签名如下? v
代码如下: std::vector s; s.push_back(~Dword(0)); 什么是~Dword?它是如何工作的? 最佳答案 Dword 这里是一个数字类型(可能是 DWORD 的类型别名
我正在测试 C++ 中推回对象与推回对象指针到 Vector 之间的性能差异。 我在 Stackoverflow 和其他文章中读到,您应该避免向后推指针,除非您必须这样做... 但是,我意识到推回指针
关闭。这个问题是not reproducible or was caused by typos .它目前不接受答案。 这个问题是由于错别字或无法再重现的问题引起的。虽然类似的问题可能是on-topi
我有一个很大的 .txt 文件,需要加载并存储在 vector 中。该文件大小约为 5MB,500 000 行,每行约 10-20 个字符,以 '\n' 分隔。我正在使用以下示例代码对读取整个文件所需
这个问题在这里已经有了答案: What is object slicing? (18 个答案) 关闭 9 年前。 我有这两个类: class A { public: A(); vir
在我当前的代码中,我想将新的 DrawObjects 插入到我创建的 vector 中, std::vector 对象; 有什么区别: objects.push_back(DrawObject(nam
我将 Cygwin 与 GCC 一起使用,最终我想将字 rune 件读入字符 vector ,并使用此代码 #include #include #include using namespace
以下代码创建一个临时对象 A 并将其推送到一个 vector 中。 在 push_back 期间删除复制构造函数并调用移动构造函数。我不确定这段代码的设计是否正确,肯定存在内存泄漏。 #include
我知道push_back可以抛出bad_alloc异常,并且如果没有try catch block ,则调用析构函数是不正确的。如果有任何push_back抛出并且它不在try catch block
正如 Scott Meyers 所指出的(http://channel9.msdn.com/Events/GoingNative/2013/An-Effective-Cpp11-14-Sampler
我正在写一个 push_back将临时容器添加到另一个容器的函数。 并且它应该在使用 push_back 之前调整或保留容器(如果两者都可用,它应该更喜欢保留而不是调整大小) 当前代码是: names
这个问题在这里已经有了答案: Create an array when the size is a variable not a constant (2 个答案) 关闭 3 年前。 我认为我在代码中
代码: // test2.cpp #include #include struct test_class { test_class() = default; test_class(
我试图在 forloop 的 vector 中推回一个 const char*。 char_temp 是一个名为 segment 的结构,而 chars_temp 是一个结构 vector 。请参阅下
在 C++ 入门书第 (3) 章中,有以下 for 循环将 vector 中的元素重置为零。 vector ivec; //UPDATE: vector declaration for (vector
class A { public: A():a(0) {} A(int x):a(x) { coutve
我正在尝试使用 Visual Leak Detector 查找内存泄漏。它告诉我 m_neighbors.push_back(ent);导致泄漏。 (简短调用堆栈 = NeighborCalculat
我们正在制作一个包含棋盘游戏信息(名称、年份、分数)的列表。我们从 .csv 文件中扫描信息,根据该信息创建一个结构,然后将该结构添加到列表中。我们一直这样做,直到文档阅读完毕。问题是列表的 push
以下代码是将“非重叠”的 TablePath 从 vector v 移动到 vector u。我在“u.push_back(*it1);”行遇到段错误。我没有复制对象(而是只复制对象的指针)所以我相信
我是一名优秀的程序员,十分优秀!