C++的类型转换

数据类型是对内存的抽象,在实际的开发过程中,我们常常会遇到把一种类型转换成另外一种类型的情况。

那么,在C/C++中,类型转换都有哪些玩法呢?

C 的类型转换

写法:(type_name) expression

#include <stdio.h>
main() {
    double d = (double) 5;
    printf("%f\n", d);
    int i = (int) 5.4; // OK
    printf("%d\n", i);
    int i2 = (int) (5.4); // OK
    printf("%d\n", i2);
    int i3 = int (5.4); // Error
    printf("%d\n", i3);
    double result = (double) 4 / 5; // OK
    printf("%f\n", result);
}

C++的类型转换

把一种数据类型转换成另外一种数据类型,可以是显式的,也可以是隐式的。

C++中的隐式转换

隐式转换不需要任何转换运算符。

他们通常自动发生在将一个类型的值拷贝给另外一个兼容的类型时。

我们来看一个例子:

#include <iostream>
class A {
    int x = 0;
public:
    A(int x) { this->x = x; }
    int getX() { return x; }
};
class B {
    int x = 0, y = 0;
public:
    B(A a) { x = a.getX(); y = a.getX(); }
    void getXY(int &x, int &y) { x = this->x; y = this->y; }
};
int main() {
    A a(10);
    B b = a; // 隐式转换
    int x, y;
    b.getXY(x, y);
    std::cout << "x: " << x << " y: " << y << std::endl;
    return 0;
}

B b = a;处发生了隐式转换,程序输出是

x: 10 y: 10

C++ 关键词 explicit

C++11之后引入了关键词explicit,如果加了explicit, 将不允许 隐式转换复制初始化.

我们如果在上面的例子中加入explicit

explicit B(A a) { x = a.getX(); y = a.getX(); }

将会发生编译错误:

error: conversion from 'A' to non-scalar type 'B' requested

C++中的显示转换

我们可以像C一样的方式进行强制转换

B b = (B) a;
B b = B (a); 

像上面一样直接强转而不使用转换运算符,有啥坏处呢?

它可能导致代码编译通过运行出错。就像下面的例子:

#include <iostream>
class Data { float i = 5, j = 6; };
class Addition {
   int x, y;
public:
   Addition(int x, int y) { this->x = x; this->y = y; }
   int result() { return x + y; }
};
int main() {
    Data d;
    Addition *add = (Addition *) &d;
    std::cout << add->result();
    return 0;
}

几种转换运算符

传统的显式类型转换允许将任何指针转换为任何其他指针类型,而与它们指向的类型无关。这可能导致运行错误或其他意外的结果。

为了更好的控制类型转换,C++提供了四种基础的转换运算符:

static_cast, const_cast, dynamic_cast, reinterpret_cast.

static_cast

static_cast通常用于转换非多态类型

static_cast转换运算符可以用来执行任何隐式转换,包括标准类型和用户定义的类型。就像下面这样:

typedef unsigned char BYTE;  
void f() {  
   int i = 65;  
   float f = 2.5;  
   char ch = static_cast<char>(i);   // int to char  
   double dbl = static_cast<double>(f);   // float to double  
   i = static_cast<BYTE>(ch);  
} 

static_cast转换运算符可以将一个整型数据转换成枚举类型。如果这个整型数据超过了枚举定义的范围,那么转换的结果是未定义的。

enum Status {
    On = 0,
    Off
};
Status s = static_cast<Status>(3);

你也可以使用static_cast将子类转换成基类,也可以将基类转换成子类。但将基类转换成子类是不安全的。

class B {};    
class D : public B {};  
void f(B* pb, D* pd) {  
   D* pd2 = static_cast<D*>(pb);   // 不安全,D可能有B没有的数据或方法  
   B* pb2 = static_cast<B*>(pd);   // 安全
}  

const_cast

const_cast用来删除const属性。

#include <iostream>
class B {
public:
    int num = 5;
};
int main() {
    using namespace std;
    const B b1;

    b1.num = 10; // Error: assignment of member 'B::num' in read-only object
    cout << b1.num << endl;

    B b2 = const_cast<B>(b1); // Error: invalid use of const_cast with type 'B', 
                              // which is not a pointer, reference, nor a pointer-to-data-member type
    b2.num = 15;
    cout << b1.num << endl;

    B *b3 = const_cast<B *>(&b1); // OK 
    b3->num = 20;
    cout << b1.num << endl;

    B &b4 = const_cast<B &>(b1); // OK
    b4.num = 25;
    cout << b1.num << endl;
}

另外一个MSDN上的例子:

#include <iostream>
using namespace std;
class CCTest {
public:
   void setNumber( int );
   void printNumber() const;
private:
   int number;
};

void CCTest::setNumber( int num ) { number = num; }

void CCTest::printNumber() const {
   cout << "\nBefore: " << number;
   const_cast< CCTest * >( this )->number--;
   cout << "\nAfter: " << number;
}

int main() {
   CCTest X;
   X.setNumber( 8 );
   X.printNumber();
}

在第二个例子中,我们也可以使用mutable关键字来进行处理。

dynamic_cast

dynamic_cast 通常用于转换多态类型

其目的是确保类型转换的结果是所请求类的有效完整对象。

将子类转换成基类时,dynamic_cast 都是成功的。

class B { };  
class C : public B { };  
class D : public C { };  
  
void f(D* pd) {  
   C* pc = dynamic_cast<C*>(pd);   // ok: C is a direct base class  
                                   // pc points to C subobject of pd   
   B* pb = dynamic_cast<B*>(pd);   // ok: B is an indirect base class  
                                   // pb points to B subobject of pd  
} 

上面例子的这种转换我们通常叫做向上转换,向上转换是一种隐式转换。因此这种情况下,我们使用static_cast或者dynamic_cast都是一样的。

我们再来看一个向下转换(将基类转换成子类)的例子:

#include <iostream>
class B {virtual void f(){}};
class D : public B {};

int main() {
    B* b = new D;
    B* b2 = new B;

    D* d = dynamic_cast<D*>(b);
    if (d == nullptr) {
        std::cout << "null pointer on first type-casting." << std::endl;
    }
    D* d2 = dynamic_cast<D*>(b2);
    if (d2 == nullptr) {
        std::cout << "null pointer on second type-casting." << std::endl;
    }
    return 0;
}

显然,输出结果是

null pointer on second type-casting.

再来看一个子类之间转换的例子:

#include <iostream>
class A {
public:
    int num = 5;
    virtual void f(){}
};

class B : public A {};

class D : public A {};

int main() {
    B *b = new B;
    b->num = 10;
    D *d1 = static_cast<D *>(b); // Error: invalid static_cast from type 'B*' to type 'D*'
    D *d2 = dynamic_cast<D *>(b); // OK
    std::cout << (d2 == nullptr) << std::endl; // 1
    delete b;
}

一个比较复杂的例子:

class A {virtual void f();};  
class B : public A {virtual void f();};  
class C : public A { };  
class D {virtual void f();};  
class E : public B, public C, public D {virtual void f();};  
  
void f(D* pd) {  
   E* pe = dynamic_cast<E*>(pd);  
   B* pb = pe;   // upcast, implicit conversion  
   A* pa = pb;   // upcast, implicit conversion  
}  

reinterpret_cast

reinterpret_cast 用来进行强制的数据转换(内存的重新解释),不管他们转换的类型之间有没有关联。

class A {};
class B {};
A *a = new A;
B *b = reinterpret_cast<B*>(a);

误用 reinterpret_cast 操作符很容易变得不安全。 除非所需的转换本质上是低级的,否则你应该使用其他强制转换运算符。

但是,它不能删除const属性。

class A {};
int main() {
    const A a;
    A *b = reinterpret_cast<A*>(&a); // Error: reinterpret_cast from type 'const A*' to type 'A*' casts away qualifiers
}

思考一下

  1. 下面的例子中,哪些地方会有编译错误?

    class A {
    public:
        virtual void foo() { }
    };
    class B {
    public:
        virtual void foo() { }
    };
    class C : public A , public B {
    public:
        virtual void foo() { }
    };
    void bar1(A *pa) {
        B *pc = dynamic_cast<B*>(pa);
    }
    void bar2(A *pa) {
        B *pc = static_cast<B*>(pa);
    }
    void bar3() {
        C c;
        A *pa = &c;
        B *pb = static_cast<B*>(static_cast<C*>(pa));
    }
    int main() {
        return 0;
    }
    
  2. 下面的模板函数是安全的吗?如果不是,更好的做法是什么?

    template <class T>
    unsigned char* alias(T& x) {
        return (unsigned char*)(&x);
    }
    

Reference

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容