今天来学习类型转换函数。

一回顾

我们学习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修饰时只能进行显示转换:

  • 转换方式:
    1. static_cast(value);
    2. 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。转换的时候是调用这个转换构造函数进行转换的。

三总结:

  1. 转换构造函数只有一个参数
  2. 转换构造函数的参数类型是其他类型
  3. 转换构造函数在类型转换时被调用
  4. 隐式类型转换时工程中BUG的重要来源
  5. explicit关键字修饰转换构造函数,只能进行显示转换,防止隐式转换的出现,杜绝BUG

想一起探讨以及获得各种学习资源加我(有我博客中写的代码的原稿):
qq:1126137994
微信:liu1126137994
可以共同交流关于嵌入式,操作系统,C++语言,C语言,数据结构等技术问题。