C++ 构造转换函数和强制转换函数
http://blog.csdn.net/chenyiming_1990/article/details/8862497
1.对于系统的预定义基本类型数据,C++提供了两种类型转换方式:隐式类型转换和显式类型转换。
- int a=5,sum;
- double b=5.55;
- sum=a+b;//-------(1)
- std::cout<<"隐式转换:a+b="<<sum<<std::endl;
- sum=(int)(a+b);//-------(2)
- sum=int(a+b);//-------(3)
- std::cout<<"显式转换:a+b="<<sum<<std::endl
上述代码中的(1)就是含有隐式类型转换的表达式,在进行"a+b"时,编译系统先将a的值5转换为双精度double,然后和b相加得到10.55,在向整形变量sum赋值时,将10.55转换为整形数10,赋值为变量sum。这种转换是C++编译系统自动完成,不需要用户去干预。而上例中的(2)和(3)中则涉及到了显式类型转换,它们都是把a+b所得结果的值,强制转化为整形数。只是(2)式是C语言中用到的形式:(类型名)表达式,而(3)式是C++中的采用的形式:类型名(表达式);
2.那么对于用户自定义的类类型而言,有该如何去实现它们和其他数据类型之间的转换呢,C++中提供了两种方法:
(1)通过转换构造函数进行类型转换;
(2)通过类型转换函数进行类型转换;
毫无疑问转换构造函数就是构造函数的一种,只不过它拥有类型转换的作用罢了。是否记得在C++之运算重载符(1)中两个复数(sum=com1+com2)相加的实例,现在如果我想要实现sum=com1+5.5,那该怎么办,也许你首先会想到再定义一个关于复数加双精度的运算符重载函数。这样做的确可以。另外我们还可以定义一个转换构造函数来解决上述的问题。我们对Comlex类(复数类)进行这样改造:
- #include <iostream>
- class Complex //复数类
- {
- private://私有
- double real;//实数
- double imag;//虚数
- public:
- Complex(double real,double imag)
- {
- this->real=real;
- this->imag=imag;
- }
- Complex(double d=0.0)//转换构造函数
- {
- real=d;//实数取double类型的值
- imag=0;//虚数取0
- }
- Complex operator+(Complex com1);//或friend Complex operator+(Complex com1,Complex com2);
- void showComplex();
- };
- Complex Complex::operator+(Complex com1)
- {
- return Complex(real+com1.real,imag+com1.imag);
- }
- void Complex::showComplex()
- {
- std::cout<<real;
- if(imag>0)
- std::cout<<"+";
- if(imag!=0)
- std::cout<<imag<<"i"<<std::endl;
- }
- int main()
- {
- Complex com(10,10),sum;
- sum = com + 5.5;//5.5调用Complex(5.5)生成临时对象,它的复数是5.5+0i
- sum.showComplex();//输出运算结果
- return 0;
- }
结果:
15.5 + 10i
上述代码在执行5.5时,调用了转换构造函数,将double类型的5.5转换为无名的Complex类的临时对象(5.5+0i),然后执行两个Complex类(复数类)对象相加的运算符重载函数。所以说一般的转换构造函数的定义形式:
类名(待转换类型)
{
函数体;
}
转换构造函数不仅可以将预定义的数据类型转换为自定义类的对象,也可以将另一个类的对象转换成转换构造函数所在的类的对象。
转换构造函数可以把预定义类型转化为自定义类的对象,但是却不能把类的对象转换为基本数据类型。比如:不能将Complex类(复数类)的对象转换成double类型数据。于是在C++中就用类型转换函数来解决这个问题。定义类型转换函数一般形式:
operator 目标类型()
{
...
return 目标类型的数据;
}
目标类型是所要转化成的类型名,既可以是预定义及基本类型也可以是自定义类型。
类型转换函数的函数名(operator 目标类型)前不能指定返回类型
没有参数
但在函数体最后一条语句一般为return语句,返回的是目标类型的数据。
现在我们对Complex类做类似改造:
- #include <iostream>
- class Complex //复数类
- {
- private://私有
- double real;//实数
- double imag;//虚数
- public:
- Complex(double real,double imag)
- {
- this->real=real;
- this->imag=imag;
- }
- Complex(double d=0.0)//转换构造函数
- {
- real=d;//实数取double类型的值
- imag=0;//虚数取0
- }
- Complex operator+(Complex com1);//或friend Complex operator+(Complex com1,Complex com2);
- void showComplex();
- operator double();
- };
- Complex Complex::operator+(Complex com1)
- {
- return Complex(real+com1.real,imag+com1.imag);
- }
- void Complex::showComplex()
- {
- std::cout<<real;
- if(imag>0)
- std::cout<<"+";
- if(imag!=0)
- std::cout<<imag<<"i"<<std::endl;
- }
- Complex::operator double()
- {
- return real;
- }
- int main()
- {
- Complex com(10,10);
- double total1,total2;
- total1 = double(com) + 5.5;//double(com)把复数(10+10i)转换为双精度数10.0
- total2 = 5.5 + com; // 写成 total2 = com + 5.5 是错误的
- std::cout<<"把Complex类显式对象转化为double类型与5.5相加为:"<< total1 << std::endl;
- std::cout<<"把Complex类对象隐式转化为double类型与5.5相加为:"<< total2 << std::endl;
- return 0;
- }
说明:
total2 = 5.5 + com; com自动调用转换函数进行转换(如果5.5 改为5,并且还定义有operator long() 类型转换,将出现编译错误,因为int 既可以转换为long ,也可以转换为 double,编译器会报错)。
为什么 total2 = com + 5.5; 不行呢?
这是会出现歧义:
是com 用转换函数变为double型,进行double定义的加法
还是 5.5 利用转换构造函数变为Complex临时变量,进行 Complex定义的加法,就不清楚了
所以进行显示的转换:
- total = (double)com + 5.5
3.最后对类型转换函数做几点补充:
(1)类型转换函数只能作为类的成员函数,不能定义为类的友元函数;
(2)类型转换函数中必须有return语句,即必须送回目标类型的数据作为函数返回值;
(3)一个类可以定义多个类型转换函数,C++编译器会根据函数名来自动调用相应的类型转换函数函数
C++ 构造转换函数和强制转换函数的更多相关文章
- JAVA数据类型自动转换,与强制转换
一.数据类型自动转换 public class Test{ public static void main(String[] args){ int a = 1; double b = 1.5; dou ...
- Number 强制类型转换 int 强制转换整型 float 强制转换浮点型 complex 强制转换成复数 bool 强制转换成布尔类型,结果只有两种,要么True 要么 False """bool 可以转换所有的数据类型 everything"""
# ###Number 强制类型转换 var1 = 5 var2 = 4.85 var3 = True var3_2 = False var4 = 3+9j var5 = "888777&q ...
- 编写高质量代码改善C#程序的157个建议[正确操作字符串、使用默认转型方法、却别对待强制转换与as和is]
前言 本文主要来学习记录前三个建议. 建议1.正确操作字符串 建议2.使用默认转型方法 建议3.区别对待强制转换与as和is 其中有很多需要理解的东西,有些地方可能理解的不太到位,还望指正. 建议1. ...
- [Effective JavaScript 笔记]第59条:避免过度的强制转换
js是弱类型语言.许多标准的操作符和代码库会把输入参数强制转换为期望的类型而不是抛出错误.如果未提供额外的逻辑,使用内置操作符的程序会继承这样的强制转换行为. functin square(x){ r ...
- C#高级编程9-第7章 运算符和类型强制转换
运算符和类型强制转换 1.运算符 运算符的简化操作 条件运算符: if-else的简化操作,也称三元运算符.如果条件为真,返回一个值,为假返回另外一个值. condition?true_value:f ...
- C#程序编写高质量代码改善的157个建议[正确操作字符串、使用默认转型方法、却别对待强制转换与as和is]
前言 本文主要来学习记录前三个建议. 建议1.正确操作字符串 建议2.使用默认转型方法 建议3.区别对待强制转换与as和is 其中有很多需要理解的东西,有些地方可能理解的不太到位,还望指正. 建议1. ...
- 3_PHP表达式_5_数据类型转换_类型强制转换
以下为学习孔祥盛主编的<PHP编程基础与实例教程>(第二版)所做的笔记. PHP类型转换分为类型自动转换和类型强制转换. 3.5.2 类型强制转换 类型强制转换允许编程人员手动将变量的数据 ...
- 如何:使用 as 和 is 运算符安全地进行强制转换(C# 编程指南)
如何:使用 as 和 is 运算符安全地进行强制转换(C# 编程指南) 由于对象是多态的,因此基类类型的变量可以保存派生类型.若要访问派生类型的方法,需要将值强制转换回该派生类型.不过,在这些情况下, ...
- C#中 As 和强制转换的总结
1.1.1 摘要 C#是一门强类型语言,一般情况下,我们最好避免将一个类型强制转换为其他类型,但有些时候难免要进行类型转换. 先想想究竟哪些操作可以进行类型转换(先不考虑.NET提供的Parse),一 ...
随机推荐
- sql语句之group_concat函数
MySQL中group_concat函数 完整的语法如下: group_concat([DISTINCT] 要连接的字段 [Order BY ASC/DESC 排序字段] [Separator '分隔 ...
- JS框架设计之模块加载系统
任何语言一到大规模应用阶段,必然要拆封模块,有利于维护和团队协作,与Java走得最近的dojo率先引进了加载器,使用document.write与同步Ajax请求实现,后台dojo以JSONP的方法来 ...
- oracle 层次化查询(生成菜单树等)
1.简介:Oracle层次化查询是Oracle特有的功能实现,主要用于返回一个数据集,这个数据集存在树的关系(数据集中存在一个Pid记录着当前数据集某一条记录的Id). 2.层次化查询主要包含两个子句 ...
- 案例19-页面使用ajax显示类别菜单
1 版本一 版本只能在首页显示类别,当切换到了其它页面就不会显示 1 web层IndexServlet代码 package www.test.web.servlet; import java.io.I ...
- 【CSS】 布局之浮动float和绝对定位absolute的选择
浮动float: 浮动的框可以向左或向右移动,直到它的外边缘碰到包含框或另一个浮动框的边框为止. 由于浮动框不在文档的普通流中,所以文档的普通流中的块框表现得就像浮动框不存在一样.(W3C) 绝对定位 ...
- 在 Linux 平台下使用 JNI
引言 Java 的出现给大家开发带来的极大的方便.但是,如果我们有大量原有的经过广泛测试的非 Java 代码,将它们全部用 Java 来重写,恐怕会带来巨大的工作量和长期的测试:如果我们的应用中需要访 ...
- 入门系列之在Nginx配置Gzip
欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文由小铁匠米兰的v 发表于云+社区专栏 简介 网站加载的速度取决于浏览器必须下载的所有文件的大小.减少要传输的文件的大小可以使网站不仅加载 ...
- C# WinForm API 改进单实例运行
在普通的单实例中,第二次点击软件快捷方式的时候,往往简单提示"系统已经运行",而不是把第一次打开的软件主窗体显示出来,下面演示如果主窗体已经打开则把第一次打开的主窗体放置到最前面; ...
- Swift数据类型_整型和浮点型
//swift中的整型和浮点型 /** * //类型推断整数是Int 浮点数是Double ,日常使用需要注意不能越界,存储时间毫秒数 英雄经验数等等之类内容容易越界 整型 大多数情况下,你不需要在代 ...
- .Net Core GB2312编码问题
1.今天抓取了一个网页的源代码.发现中文是乱码的,马上第一反应是编码问题..... 2.仔细一看基于WebClient写的代码,还真的是没有设置编码... /// <summary> // ...