从initializer_list构造 std::map<T、<S>unique_ptr> 时出错

Error Constructing std::map<T, unique_ptr<S>> from an initializer_list

本文关键字:gt lt initializer unique ptr 出错 map 构造 list std      更新时间:2023-10-16

C2280 - 尝试引用已删除的函数 是我尝试从initializer_list构造 std::map> 时遇到的错误

我已经制作了最简单的代码(如下)来表示我的问题:

#include <string>
#include <map>
#include <memory>
using namespace std;
int main() {
    map<string, unique_ptr<int>> MyMap{
        { "first" ,make_unique<int>(6)  },
        { "second",make_unique<int>(22) },
        { "third" ,make_unique<int>(86) }
    };
}

所以我意识到unique_ptr无法复制,但是既然make_unique返回了一个右值,那么unique_ptr不应该被移动到initializer_list吗?顺便说一句,如果我用shared_ptr替换unique_ptr,那么问题就解决了,但不是我正在寻找的解决方案,我想保留该初始化样式和unique_ptr。那么,在什么时候,对已删除函数的这种有问题的调用正在发生呢?有没有办法在不替换unique_ptr的情况下编译它?

可悲的是,这是std::initializer_list对象的局限性之一。它们仅提供对其元素的const访问,因此无法移出它们。最好的办法是以老式的方式初始化地图。

map<string, unique_ptr<int>> MyMap;
MyMap["first"] = make_unique<int>(6);
// etc.

如果您需要const地图,您可以创建一个临时地图并从中移动:

const map<string, unique_ptr<int>> myMap = [] {
    map<string, unique_ptr<int>> m;
    m["first"] = make_unique<int>(6);
    // etc.
    return m;  // RVO or implicit move
}();

你也可以使用 std::map 的范围构造函数,但是你必须自己为此编写一个特殊的迭代器类型,这会很烦人。