C++ 语言隐式的类类型转换 - explicit 构造函数

C++ 语言隐式的类类型转换 - explicit 构造函数

如果构造函数只接受一个实参,则它实际上定义了转换为此类类型的隐式转换机制,我们把这种构造函数称作转换构造函数 (converting constructor)。能通过一个实参调用的构造函数定义了一条从构造函数的参数类型向类类型隐式转换的规则。

转换构造函数 (converting constructor) 可以用一个实参调用的非显式构造函数。这样的函数隐式地将参数类型转换成类类型。

Sales 类中,接受 std::string 的构造函数和接受 std::istream 的构造函数分别定义了从这两种类型向 Sales 隐式转换的规则。在需要使用 Sales 地方,我们可以使用 std::string 或者 std::istream 作为替代:

std::string null_book = "9-99999-9";
// 构造一个临时的 Sales 对象
// 该对象的 sold_num 和 revenue 等于 0,book_no 等于 null_book
item.combine(null_book);

在这里我们用一个 std::string 实参调用了 Salescombine 成员。该调用是合法的,编译器用给定的 std::string 自动创建了一个 Sales 对象。新生成的这个 (临时) Sales 对象被传递给 combine。因为 combine 的参数是一个常量引用,所以我们可以给该参数传递一个临时量。

1. 只允许一步类类型转换

编译器只会自动地执行一步类型转换。因为下面的代码隐式地使用了两种转换规则,所以是错误的:

// 错误:需要用户定义的两种转换
// (1) 把 "9-99999-9" 转换成 std::string
// (2) 再把这个 (临时的) std::string 转换成 Sales
item.combine("9-99999-9");

如果我们想完成上述调用,可以显式地把字符串转换成 std::string 或者 Sales 对象:

// 正确:显式地转换成 std::string,隐式地转换成 Sales
item.combine(std::string("9-99999-9"));

// 正确:隐式地转换成 std::string,显式地转换成 Sales
item.combine(Sales("9-99999-9"));

2. 类类型转换不是总有效

null_book 中的 std::string 可能表示了一个不存在的 ISBN 编号。

std::istreamSales 的转换如下:

// 使用 std::istream 构造函数创建一个函数传递给 combine
item.combine(cin);

这段代码隐式地把 std::cin 转换成 Sales,这个转换执行了接受一个 std::istreamSales 构造函数。该构造函数通过读取标准输入创建了一个 (临时的) Sales 对象,随后将得到的对象传递给 combine

Sales 对象是个临时量,一旦 combine 完成我们就不能再访问它了。实际上,我们构建了一个对象,先将它的值加到 item 中,随后将其丢弃。

3. 抑制构造函数定义的隐式转换

在要求隐式转换的程序上下文中,我们可以通过将构造函数声明为 explicit 加以阻止:

class Sales {
public:
	Sales() = default;
	Sales(const std::string &s, unsigned int n, double p) : book_no(s), sold_num(n), revenue(p*n) {}
	explicit Sales(const std::string &s) : book_no(s) {}
	explicit Sales(std::istream &);
	// ......
};

没有任何构造函数能用于隐式地创建 Sales 对象,之前的两种用法都无法通过编译:

item.combine(null_book);  // 错误:std::string 构造函数是 explicit 的
item.combine(cin);  // 错误:std::istream 构造函数是 explicit 的

构造函数声明为 explicit,从而可以避免我们不期望的隐式转换。

关键字 explicit 只对一个实参的构造函数有效。需要多个实参的构造函数不能用于执行隐式转换,所以无须将这些构造函数指定为 explicit 的。只能在类内声明构造函数时使用 explicit 关键字,在类外部定义时不应重复:

// 错误:explicit 关键字只允许出现在类内的构造函教声明处
explicit Sales::Sales(std::istream& is)
{
	read(is, *this);
}

4. explicit 构造函数只能用于直接初始化

发生隐式转换的一种情况是当我们执行拷贝形式的初始化时 (使用 =)。此时,我们只能使用直接初始化而不能使用 explicit 构造函数:

Sales iteml(null_book);  // 正确:直接初始化
// 错误:不能将 explicit 构造函数用于拷贝形式的初始化过程
Sales_data item2 = null_book;

当我们用 explicit 关键字声明构造函数时,它将只能以直接初始化的形式使用。而且,编译器将不会在自动转换过程中使用该构造函数。

显式构造函数 (explicit constructor) 是可以用一个单独的实参调用但是不能用隐式转换的构造函数。通过在构造函数之前加 explicit 关键字就可以将其声明成显式构造函数。

5. 为转换显式地使用构造函数

尽管编译器不会将 explicit 的构造函数用于隐式转换过程,但是我们可以使用这样的构造函数显式地强制进行转换:

// 正确:实参是一个显式构造的 Sales 对象 
item.combine(Sales(null_book));
// 正确:static_cast 可以使用 explicit 的构造函数 
item.combine(static_cast<Sales_data> (cin));

在第一个调用中,我们直接使用 Sales 的构造函数,该调用通过接受 std::string 的构造函数创建了一个临时的Sa1es 对象。在第二个调用中,我们使用 static_cast 执行了显式的而非隐式的转换。其中,static_cast 使用 std::istream构造函数创建了一个临时的 Sales 对象。

6. 标准库中含有显式构造函数的类

我们用过的一些标准库中的类含有单参数的构造函数:

  • 接受一个单参数的 const char*std::string 构造函数不是 explicit 的。
  • 接受一个容量参数的 vector 构造函数是 explicit 的。

实现 (implementation) 类的成员 (通常是私有的),定义了不希望为使用类类型的代码所用的数据及任何操作。

References

(美) Stanley B. Lippman, (美) Josée Lajoie, (美) Barbara E. Moo 著, 王刚, 杨巨峰 译. C++ Primer 中文版[M]. 第 5 版. 电子工业出版社, 2013.
https://www.informit.com/store/c-plus-plus-primer-9780321714114

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Yongqiang Cheng

梦想不是浮躁,而是沉淀和积累。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值