无法<string>从"常量字符 []"转换为<类名>

could not convert <string> from ‘const char []’ to <class name>

本文关键字:gt lt 转换 类名 字符 无法 常量 string      更新时间:2023-10-16

>我有一个问题,它构成了以下程序:

class Player {
public:
Player(int new_id);
private:
string name;
int id;
};
class Team {
public:
Team(string new_name);
private:
Player* player;
string name;
};
int main()
{
Team a("Team1");
}
Team::Team(string new_name)
{
name=new_name;
player= new Player[2]{1,2};
}
Player::Player(int new_id)
{
id=new_id;
}

该程序有效,但是当我将其更改为此程序时:

class Player {
public:
Player(string new_name);
private:
string name;
int id;
};
class Team {
public:
Team(string new_name);
private:
Player* player;
string name;
};
int main()
{
Team a("Team1");
}
Team::Team(string new_name)
{
name=new_name;
player= new Player[2]{"Bob","Tom"};
}
Player::Player(string new_name)
{
name=new_name;
}

它给了我这个错误:无法将"鲍勃"从"常量字符 [4]"转换为"播放器"。我不明白。我唯一改变的是我使用字符串而不是 int。 有什么想法吗? 提前感谢!

您的问题是关于用户定义的转换,如C++标准所指定:

转化 [类.conv]

类对象的类型转换可以由构造函数指定,并且 通过转换函数。这些转换称为用户定义 转换,用于隐式类型转换(条款 7(,用于 初始化 (11.6( 和显式类型转换 (8.4、8.2.9(。

问题是:

。最多一个用户定义的转换(构造函数或转换( 函数(隐式应用于单个值。

在您的工作示例中:

player= new Player[2]{1,2};

您有一个用户定义的转换在这里发生:int通过使用采用int参数的Player的构造函数进行Player

但在另一种情况下:

player= new Player[2]{"Bob","Tom"};

在这里,您需要进行两个用户定义的转换:使用其构造函数到std::string的文本 char 字符串,然后是采用std::string参数的Player的构造函数。此处需要两个用户定义的转换。C++标准最多允许一个。失败。

我只能想到两种可能的解决方法。一种是手动消除其中一个用户定义的转换:

player= new Player[2]{std::string{"Bob"}, std::string{"Tom"}};

现在只剩下一个用户定义的转换,使用其构造函数从std::stringPlayer

第二种解决方法是作弊并Player使用模板构造函数:

class Player {
public:
template<typename Arg>
Player(Arg &&arg)
: name{std::forward<Arg>(arg)}
{
}
private:
std::string name;
int id;
};

这有效地接受Player构造函数的任何参数,将用户定义的转换之一向下戳到字符串的构造函数。我想,可以在这里进行一些改进,并且只有在std::string的构造成功时,才能使该构造函数参与重载解析。这将需要更多的工作。

问题是在这个语句中

player= new Player[2]{"Bob","Tom"};

需要两个用户定义的隐式转换。第一个是将字符串文本转换为类型std::string,第二个是将类型为std::tring的对象转换为 Player 类型。这是C++标准所不允许的。

若要避免此错误,请向类再添加一个构造函数。例如

class Player {
public:
Player( const std::string &new_name) : name( new_name ) {}
Player( const char *new_name) : name( new_name ) {}
private:
string name;
int id;
};

或者不添加新的构造函数,你可以只写

player= new Player[2]{ Player( "Bob" ), Player( "Tom" ) };