std::仅移动类型列表:不能在 VC++ 中放入 std::vector
std::list of move-only type: Cannot emplace into std::vector in VC++
在VC++ 2019中,我无法emplace_back
仅移动类型的(右值)list
。
#include <vector>
#include <list>
struct A
{
A(A&&) {}
};
using ListOfA = std::list<A>;
int main()
{
std::vector<ListOfA> v;
// Build error in VC++ 2019
// No error in Clang and GCC C++11 - C++2a
v.emplace_back(std::move(ListOfA()));
}
尝试在 VC++ 2019 中构建会给出以下编译错误:
'A::A(const A &)': attempting to reference a deleted function
显然,VC++正在尝试实例化A
的(左值)复制构造函数,该构造函数(正确)不存在,因为我已显式定义了A
的一个构造函数。
我认为通过从另一个list
移动来就地实例化vector
中的list
应该是有效的 - 也就是说,list
类确实有一个移动构造函数,我认为它应该只是让新list
对(移自)list
中的元素拥有所有权, 不需要任何副本。
事实上,使用Wandbox,相同的代码使用GCC和Clang构建和运行而不会出错。
有人可以解释为什么这段代码不能在VC++ 2019中编译吗? 我是否有误解 - 实际上是否有正当理由为什么(左值)复制构造函数由上面的代码中的 VC++ 编译器实例化?
注意
当std::move(...)
不存在时,VC++ 中会发生相同的错误;即以下行也发生相同的错误:
v.emplace_back(ListOfA());
MSVC 使用std::list
的复制构造函数,因为它的移动构造函数正在抛出。在重新分配期间,如果移动构造函数引发,std::vector
无法提供标准要求的强异常保证。
在您的情况下,向量在重新分配之前没有任何元素,因此似乎没有调用复制构造函数,但这并不意味着不需要复制构造函数。
libstdc++和libc++中的std::list
noexcept
移动构造函数。这是允许的,但标准没有要求。
相关文章:
- 使用std::vector的OpenCL矩阵乘法
- POCO::PostgreSQL:如何将std::vector支持添加到`Binder::bind`
- std::vector的包装器,使数组的结构看起来像结构的数组
- 编译器如何区分std::vector的构造函数
- 使用 pqxx 将 std::vector 存储在 postgresql 中,并从数据库中检索它
- 在std::vector上存储带有模板的类实例
- 在main()之外初始化std::vector会导致性能下降(多线程)
- 为什么std::vector比数组慢
- std::vector::迭代器是否可以合法地作为指针
- 如何将二进制格式的 C++ 对象的 std::vector 保存到磁盘?
- 为什么std::vector和std::valarray初始化构造函数不同
- ";结果类型必须是可从输入范围的值类型""构造的;创建std::vector时
- 在没有未定义行为的情况下实现类似std::vector的容器
- 如何调整 std::vector of Eigen::MatrixXd 的大小
- 使用 std::vector::reverse_iterator 将 int 序列化为字节向量?
- 如何将AERT_Allocate与 std:vector 一起使用
- 推导 std::vector::back() 的返回类型
- 如何将原始字节附加到 std::vector?
- std::vector 没有重载函数的实例与参数列表匹配
- 如果 KEY 是 std::list 或 std::vector 而不是值,那么 std::map 的默认行为是什么?