显式转换操作符
优质
小牛编辑
126浏览
2023-12-01
C++98标准提供隐式和显式两种构造函数,也就是说,声明为显式形式的构造函数所定义的转换只能用于显式转换,而其他形式的构造函数则用于隐式转换。例如:
struct S { S(int); }; // “普通构造函数”默认是隐式转换
S s1(1); // ok, 直接构造
S s2 = 1; // ok, 隐式拷贝构造
void f(S);
// 能通过编译(但是经常会产生意外结果——如果S是vector类型会怎么样呢?)
// 译注:详见下一用例的解释
f(1);
struct E { explicit E(int); }; // 显式构造函数
E e1(1); // ok
E e2 = 1; // 错误(但是常常会让人感到意外——这怎么会错呢?)
void f(E);
// 该处会产生编译错误(而非编译通过),以避免因隐式类型转换而得到莫名其妙的结果。
// 例如std::vector::vector(int size), 该构造函数在标准库中定义为显式类型转换,
// (译注:以避免程序员为了初始化一个只含有一个元素10的数组而写出如下代码:
// vector<int> vec = 10;
// 而实际上该代码的含义却是定义一个初始包含10个元素的数组)
f(1);
然而,禁止从构造函数作隐式转换(以避免问题),并没有堵住全部漏洞。如果某个类本身禁止改动,那么可以从另一个不同的类中定义一个转换操作符。例如:
struct S { S(int) { } /* … */ };
struct SS {>
int m;
SS(int x) :m(x) { }
// 在struct S无须定义S(SS)——所谓的“非侵入”式做法
operator S() { return S(m); }
};
SS ss(1); // ok; 默认构造函数
S s1 = ss; // ok; 隐式转换为S后调用拷贝构造函数
S s2(ss); // ok; 隐式转换为S后调用直接构造函数
void f(S);
f(ss); // ok; 隐式转换为S后传参
(译注:这段代码的意义,实际是通过SS作为中间桥梁,将int转换为S。)
遗憾的是,C++98中无法定义”显式转换操作符”来完全禁止某个类相关的隐式转换(因为除此之外鲜有用武之地)。C++11则高瞻远瞩,添加了这个特性。例如:
struct S { S(int) { } };
struct SS {
int m;
SS(int x) :m(x) { }
// 因为结构体S中没有定义构造函数S(SS)
// 无法将SS转换为S,所以只好在SS中定义一个返回S的转换操作符,
// 将自己转换为S。
// 转换动作,可以由目标类型S提供,也可以由源类型SS提供。)
explicit operator S() { return S(m); }
};
SS ss(1); // ok; 默认构造函数
S s1 = ss; // 错误; 拷贝构造函数不能使用显式转换
S s2(ss); // ok; 直接构造函数可以使用显式转换
void f(S);
f(ss); // 错误; 从SS向S的转换必须是显式的.
// 译注: 强制类型转换也可使用显式转换,例如
// S s3 = static_cast<S>(ss);
参考:
- Standard: 12.3 Conversions
[N2333=07-0193] Lois Goldthwaite, Michael Wong, and Jens Maurer: