动机
只有一个参数的构造函数存在隐式类型转换的可能。比如如下构造函数:
struct Sales_data{
Sales_data() = default;
Sales_data(const string& s):bookNo(s){} //编译器也劝加上explicit,Single-argument constructors must be marked explicit to avoid unintentional implicit conversions
Sales_data &combine(const Sales_data &other){}
string bookNo;
unsigned units_sold = 0;
double revenue = 0;
};
下列情况存在string到Sales_data的隐式转换。
string null_book = "99999";
item.combine(null_book); //这里有个string到Sales_data的构造,临时量是const的
编译器只允许一步转换
如果到类类型的转换需要多步,编译器直接报错。
item.combine("99999"); //xxx,这里有个字符串到string到Sales_data转换,直接报错
item.combine(string("99999")); //这俩正确,里面只涉及一步转换
item.combine(Sales_data("99999"));
explicit抑制隐式调用
通过在构造函数前加explicit只能直接初始化
item.combine(null_book); //xxx,等价于Sales_data temp = null_book;
Sales_data item1 = null_book; //xxx,不允许隐式转换来调用构造函数。
Sales_data item1(null_book); //正确,只能用更标准的形式
标准库的一些explicit构造函数
1. string里接受const char*,不是
2.vector里接受容量参数的,是。这样才不能vector<int> item = 3;