列表和字符串之间的不明确构造函数<string>

Ambiguous constructor between list<string> and string

本文关键字:lt string gt 构造函数 不明确 字符串 之间 列表      更新时间:2023-10-16

在下面的代码中,当我试图将列表传递给构造函数时,编译器给了我一个错误:

#include <string>
#include <iostream>
#include <list>

class MyClass {
    std::list<std::string> strings;
public:
    void disp() {
        for (auto &str : strings)
            std::cout << str << std::endl;
    }
    MyClass(std::string const &str)
        : strings({str}) {}
    MyClass(std::list<std::string> const &strlist)
        : strings(strlist) {}
};

int main ()
{
    // Compiles well:
    MyClass c1("azerty");
    c1.disp();
    // Compilation error, "call to constructor of 'MyClass' is ambiguous":
    MyClass c2({"azerty", "qwerty"});
    c2.disp();
    return 0;
}

我试图将explicit添加到构造函数的声明中,但它没有改变任何东西。

问题是string有这样的构造函数:

template< class InputIt >
basic_string( InputIt first, InputIt last, 
              const Allocator& alloc = Allocator() );

{"azerty", "qwerty"}是一个不幸的匹配,因为const char*实际上是一个输入迭代器…即使两个实参不打算作为迭代器,也不是同一容器的迭代器。

一个解决方案是只提供一个构造函数,它接受一个初始化列表,并且只使用那个初始化列表:

MyClass(std::initializer_list<std::string> il)
    : strings(il.begin(), il.end())
{ }

避免这种歧义调用的一种方法是这样做,

     std::list<std::string> strList{"azerty", "qwerty"};
     MyClass c2(strList);
不是

     MyClass c2({"azerty", "qwerty"});

正如人们在评论中所说,这是因为std::string的一个构造函数重载有一个初始化列表。

可以通过添加额外的大括号来修复它:

MyClass c2({{"azerty"}, {"qwerty"}});

当我在寻找可读性时,我找到了一个解决方法,通过这样改变字符串构造函数:

 MyClass(const char *str) : strings({str}) {}

现在我可以做这两种结构:

MyClass c1("foobar");
MyClass c2({"azerty", "qwerty"});