今天来学习类型转换函数。
一回顾
我们学习C语言的时候,都知道,标准的数据类型之间,是可以做类型的转换的。分为隐式类型转换和强制类型转换。那么强制类型转换呢,比较简单。但是隐式类型转换,却存在诸多问题。下面我们看一下隐式类型转换,是如何转换的:
转换规则如下:
隐式类型转换只能是小类型转换为大类型,这样可以避免数据的损坏不完整。上图显示的就是,只能朝着箭头方向进行隐式的类型转换,而不能反过来进行隐式转换。
下面还是先分析一个例子:
#include <iostream>
#include <string>
using namespace std;
int main()
{
short s = 'a'; //char类型可以隐式的转换为short类型
unsigned int ui = 1000; //int类型可以隐式的转换为unsigned int类型。
int i = -2000;
double d = i; //int型可以隐式的转换为double类型
cout << "d = " << d << endl;
cout << "ui = " << ui << endl;
cout << "ui + i = " << ui + i << endl;//这里我们预期应该是得到负的值,但是结果却不是
if( (ui + i) > 0 )
{
cout << "Positive" << endl;
}
else
{
cout << "Negative" << endl;
}
cout << "sizeof(s + 'b') = " << sizeof(s + 'b') << endl;
/* 这里s为short类型2字节,‘b’为char类型1字节,做加法运算后,char类型为什么没有隐式转换 * short类型呢?而是将这两个类型都转换为int类型?因为编译器认为int类型的计算是 * 最为高效的,而short类型与char类型转换为int类型又是安全可行的,所以编译器就直接把 * 他们转换为int类型进行计算 */
return 0;
}
运行结果为:
d = -2000
ui = 1000
ui + i = 4294966296
Positive
sizeof(s + ‘b’) = 4
分析:
程序的分析,上面已经写的很清楚了。从此也可以看出隐式转换存在一定的隐患。
二再论构造函数
问题:
普通类型与类类型之间能否进行类型转换,类类型之间能否进行类型转换?
回答:
当然是可以的!!!只不过这里需要先引入一个概念:转换构造函数。那么什么是转换构造函数呢?
我们知道,构造函数可以有不同类型的参数,当参数满足下列条件时,成为转换构造函数:
- 有且仅有一个参数
- 参数是基本类型
- 参数是其他类型
那么转换构造函数,有什么意义呢?在说明转换构造函数之前,我们看一下下图:
上图,说明,C方式中的旧标准普通类型之间进行的强制类型转换的格式,以及类似的类类型与普通类型之间的转换的格式。上图类类型的转换,实际上是属于隐式类型的转换(可以详细看一下视频课程中的逐步索引,这里我就直接给出结论了),但是隐式类型转换呢,在工程开发中是存在诸多隐患的,是程序中BUG的重要来源。
所以我们一般在开发中,都不使用隐式转换,而使用显示转换:
转换构造函数被explicit修饰时只能进行显示转换:
- 转换方式:
- static_cast(value);
- ClassName(value); //不推荐
说了这么多,还是直接写代码分析吧:
#include <iostream>
#include <string>
using namespace std;
class Test
{
int mValue;
public:
Test()
{
mValue = 0;
}
explicit Test(int i) //转换构造函数
{
mValue = i;
}
Test operator + (const Test& p)
{
Test ret(mValue + p.mValue);
return ret;
}
int value()
{
return mValue;
}
};
int main()
{
Test t;
t = static_cast<Test>(5); // t = Test(5);显示转换
cout << t.value() << endl;
Test r;
r = t + static_cast<Test>(10); // r = t + Test(10); 显示转换
cout << r.value() << endl;
return 0;
}
运行结果为:
5
15
分析与结论:
上面的程序,可以看出,当我们想让普通类型int类型,与类类型的值进行运算时,首先需要把int类型转换为类类型(转化为一个临时的类类型)然后再进行运算。类类型的转换,需要显示进行,前面加上前缀:static_cast<calss>(int)
,就可以进行转换了,同时呢,转换还需要一个转换构造函数,转换构造函数前面也需要加上前缀:explicit。转换的时候是调用这个转换构造函数进行转换的。
三总结:
- 转换构造函数只有一个参数
- 转换构造函数的参数类型是其他类型
- 转换构造函数在类型转换时被调用
- 隐式类型转换时工程中BUG的重要来源
- explicit关键字修饰转换构造函数,只能进行显示转换,防止隐式转换的出现,杜绝BUG
想一起探讨以及获得各种学习资源加我(有我博客中写的代码的原稿):
qq:1126137994
微信:liu1126137994
可以共同交流关于嵌入式,操作系统,C++语言,C语言,数据结构等技术问题。