类型转换运算符

class SmallInt {
public: SmallInt(int i = 0) : val(i) {
if (i < 0 || i > 255)
throw std::out_of_range("Bad SmallInt value");
}
operator int() const { return val; }
private:
size_t val;
};

在这个类中,类型转换运算符为

operator int() const { return val; }

类型转换的形式为:
operator Type() const;

Type表示除了void之外任意类型,只要该类型能作为函数的返回类型。因此,不允许转换成数组或函数。但是可以转换为指针或引用类型。

类型转换函数既没有显示的返回类型, 也没有形参,必须定义为成员函数。

编译器值能执行一次进一步转换,但是我们将SmallInt转换为int后,可以将int转换为任何其他算术类型。

尽管类型转换函数不负责指定返回类型,但实际上每个类型转换函数都会返回一个对应类型的值:
class SmallInt {
public:
int operator int() const; // 错误, 指定了返回类型
operator int(int = 0) const; // 错误, 参数列表不为空
operator int*() const { return 42; } // 错误,42不是一个指针
}

显示的类型转换运算符


在C++11里,加入了显示的类型转换运算符:
class SmallInt {
public: SmallInt(int i = 0) : val(i) {
if (i < 0 || i > 255)
throw std::out_of_range("Bad SmallInt value");
}
explicit operator int() const { return val; }
private:
size_t val;
};

添加explicit后,编译器不会将一个显示的类型转换用于隐式类型转换:
SmallInt si = 3; // 构造函数不是explicit的
si + 3; // 错误, 类型转换是explicit的
static_cast<int>(si) + 3; // 正确,显示的请求类型转换

但是存在一个例外,当表达式被用作条件, 编译器才会将显示的类型转换用于它,即就是,当表达式出现在以下的位置时,显示的类型转换被隐式执行:
if、while及do语句的条件部分;
for语句头的条件表达式;
逻辑与或非运算符的运算对象
条件运算符( ? : )的条件表达式。


在C++11中,IO标准库定义了一个向bool类型的显示类型转换,因此我们可以使用IO库类型对象来判断一个流是否正确:
while (std::cin >> value) ;

当cin读完值后,由于在while的条件部分,因此显示的类型转换被隐式的执行,cin转换为一个bool类型,用于判断是否读取成功。



避免有二义性的类型转换

struct A {
A() = default;
A(const B&);
}; struct B {
operator A() const;
}; A f(const A&);
B b;
A a = f(b); // 二义性错误

上面的代码中,发生了二义性错误,调用f函数时,既可以使用以B为参数的A的构造函数,也可以使用B当中把B转换为A的类型转换运算符。效果相当,因此出现二义性。


可以显示调用函数名来执行对应的调用:
A a1 = f(b.operator A()); // 调用B的类型转换运算符
A a2 = f(A(b)); // 调用A的构造函数

尽量不要定义多重的内置类型转换,除了显示的向bool类型转换之外,应该尽量避免定义类型转换函数。

重载函数与转换构造函数

当重载函数的参数分别属于不同的类类型时,如果这些类恰好同时定义了同样的转换构造函数,那么二义性问题进一步提升:

struct C {
C(int);
}; struct D {
D(int);
}; void manip(const C&);
void manip(const D&);
manip(10); // 二义性错误,到底是manip(C(10))还是manip(D(10))呢

可以显示的调用:

manip(C(10));

重载函数与用户自定义的类型转换

struct C {
C(int);
};
struct E {
E(double);
}; void manip2(const C&);
void manip2(const E&); manip2(10); // 二义性错误,是manip2(C(10)) 还是 manip2(E(double(10)));

因为调用重载函数所请求的用户定义的类型转换不止一个且彼此不同,所以具有二义性。即使其中一个需要额外的标准类型转换而另一个调用能精确匹配,编译器也会将该调用标示为错误。


函数匹配与重载运算符


class SmallInt {
friend SmallInt operator + (const SmallInt&, const SmallInt&);
public:
SmallInt(int = 0);
operator int() const { return val; }
private:
size_t val;
}; SmallInt s1, s2;
SmallInt s3 = s1 + s2; // 使用重载的operator +
int i = s3 + 0; // 二义性错误

将s1和s2相加赋值给s3, 执行了重载的加法运算符。而最后一句有二义性错误,0可以转换成SmallInt,然后使用SmallInt的+,或者把s3转换为int,对两个int使用内置版本的+。

如果我们对同一个类既提供了转换目标是算数类型的类型转换,也提供了重载的运算符,则会遇到重载运算符与内置运算符的二义性问题。



C++ Primer : : 第十四章 : 重载运算符与类型转换之类型转换运算符和重载匹配的更多相关文章

  1. C++ Primer : 第十四章 : 重载运算与类型转换之重载运算符

    重载前须知 重载运算符是特殊的函数,它们的名字由operator和其后要重载的运算符号共同组成. 因为重载运算符时函数, 因此它包含返回值.参数列表和函数体. 对于重载运算符是成员函数时, 它的第一个 ...

  2. C++Primer 第十四章

    //1.当运算符作用于类类型运算对象时,可以通过运算符重载重新定义该运算符的含义.明智的使用运算符重载能令程序更加易于编写和阅读. //2.重载的运算符是具有特殊名字的函数,它们由关键字operato ...

  3. 【C++】《C++ Primer 》第十四章

    第十四章 重载运算与类型转换 一.基本概念 重载运算符是具有特殊名字的函数:由关键字operator和其后要定义的运算符号共同组成.也包含返回类型.参数列表以及函数体. 当一个重载的运算符是成员函数时 ...

  4. C++ Primer Plus学习:第十四章

    第十四章 C++中的代码重用 包含对象成员的类 将类的对象作为新类的成员.称为has-a关系.使用公有继承的时候,类可以继承接口,可能还有实现(纯虚函数不提供实现,只提供接口).使用包含时,可以获得实 ...

  5. C Primer Plus_第四章_字符串和格式化输入输出_编程练习

    Practice 1.输入名字和姓氏,以"名字,姓氏"的格式输出打印. #include int main(void) { char name[20]; char family[2 ...

  6. 《Linux命令行与shell脚本编程大全》 第十四章 学习笔记

    第十四章:呈现数据 理解输入与输出 标准文件描述符 文件描述符 缩写 描述 0 STDIN 标准输入 1 STDOUT 标准输出 2 STDERR 标准错误 1.STDIN 代表标准输入.对于终端界面 ...

  7. perl 第十四章 Perl5的包和模块

    第十四章 Perl5的包和模块 by flamephoenix 一.require函数  1.require函数和子程序库  2.用require指定Perl版本二.包  1.包的定义  2.在包间切 ...

  8. Gradle 1.12 翻译——第十四章. 教程 - 杂七杂八

    有关其它已翻译的章节请关注Github上的项目:https://github.com/msdx/gradledoc/tree/1.12,或訪问:http://gradledoc.qiniudn.com ...

  9. C和指针 (pointers on C)——第十四章:预处理器

    第十四章 预处理器 我跳过了先进的指针主题的章节. 太多的技巧,太学科不适合今天的我.但我真的读,读懂.假设谁读了私下能够交流一下.有的小技巧还是非常有意思. 预处理器这一章的内容.大家肯定都用过.什 ...

随机推荐

  1. Intellij自动下载导入框架包

    忽然发现intellij尽然可以自动导入 框架所需的包,而且可以选择jar包版本,瞬间发现Maven,gradle管理jar包还得写配置文件弱爆了. 以Hibernate为例: 1.ProjectSt ...

  2. 单据状态BE构建

    这节主要罗列出单据状态BE构建步骤1.创建单据状态BE实体项目,修改命名空间 2.如下图所示,分别设置实体枚举状态值 3.修改单据基本属性 构造后,至此单据状态BE构建完毕

  3. json和xml数据的解析

    一 json数据 1一条json就像一个对象,也想像OC中的数组,且内嵌了很多键值对字典 {"name" : "jack", "age" : ...

  4. vue学习笔记之属性和方法

    每个Vue都会代理其data对象里所有的属性:只有这些被代理的属性是响应的.如果在实例创建之后添加新的属性到实例上,它不会触发视图更新.例子: <script type="text/j ...

  5. 1.jquery的变量赋予方式

    1.首先通过将变量赋予 $_ = window.$ _jQuery = window.JQuery 2.将函数赋给两个全局变量 window.jQuery = window.$ = jQuery 3. ...

  6. JVM实用参数——新生代垃圾回收

    JVM实用参数目录 JVM实用参数——新生代垃圾回收 概述 第1部分  新生代垃圾回收介绍 第2部分 参数介绍 参考 第1部分  新生代垃圾回收介绍 本部分,我们将关注堆(heap) 中一个主要区域, ...

  7. maven入门探讨

    java项目最恶心的一点莫过于需要使用大量的jar.每次引用jar的时候都要自己手动去各地寻找,然后导入到项目的指定文件夹当中最后还要添加Path.这无疑是一项工作量巨大的工作,同时如果控制不当就会提 ...

  8. 【个人使用.Net类库】(4)验证码类

    验证码是现在管理系统常用的一种保护用户帐户信息的一种功能. 验证码可以有效防止某个黑客对某一个特定注册用户用特定程序暴力破解方式进行不断的登录尝试,虽然这可能是我们登录麻烦一点,但是对用户的密码安全来 ...

  9. mysq常见问题

    1.Reading table information for completion of table and column names You can turn off this feature t ...

  10. 转:Caused by: java.lang.NoSuchMethodError: org.apache.log4j.Category.log

    Caused by: java.lang.NoSuchMethodError: org.apache.log4j.Category.log出现的异常:java.lang.reflect.Invocat ...