C++类型转换

static_cast

  • 用于类层次结构中,基类和子类之间指针和引用的转换。
    • 进行上行转换,也就是把子类的指针或引用转换成父类表示,这种转换是安全的;
    • 当进行下行转换,也就是把父类的指针或引用转换成子类表示,这种转换是不安全的,也需要程序员来保证;
  • 用于基本数据类型之间的转换,如把int转换成char,把int转换成enum等等,这种转换的安全性需要程序员来保证
  • 把void指针转换成目标类型的指针,是及其不安全的;
#include "iostream";
using namespace std;

class Parent {
public:
    Parent() {
        cout << "构造Parent" << endl;
    }
    Parent(const Parent&) {
        cout << "Parent拷贝构造" << endl;
    }
    virtual void print() {
        cout << "Parent print" << endl;
    }
    void eat() {
        cout << "Parent eat" << endl;
    }
};

class Child :public Parent{
public:
    Child() {
        cout << "构造Child" << endl;
    }

    Child(const Child &c) {
        cout << "Child拷贝构造" << endl;
    }
    virtual void print() {
        cout << "Child print" << endl;
    }
    void eat() {
        cout << "Child eat" << endl;
    }
};

int main()
{
    Child c;
    Parent& p = static_cast<Parent&>(c);
    p.print();
    cout << "-------" << endl;
    //这里相当于Parent pp=c;会调用Parent的拷贝构造函数
    Parent pp = static_cast<Parent>(c);
    pp.print();
    cout << "-------" << endl;
    Parent* ppp = static_cast<Parent*>(&c);
    ppp->print();
    int n = 10;
    void * nn = &n;
    //void* 进行转换目标类型指针,可以但不安全
    int *nnn = static_cast<int *>(nn);
    cout << *nnn << endl;
    //这里编译提示报错
    //char *c = static_cast<char *>(nnn);
    return 0;
}

结果:

结果

dynamic_cast

  • dynamic_cast主要用于类层次间的上行转换和下行转换,还可以用于类之间的交叉转换。在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的;在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。
//使用前面定义的类
int main()
{
    Child c;
    Parent* p = dynamic_cast<Parent*>(&c);
    p->print();
    cout << "-----" << endl;
    Child *cc = dynamic_cast<Child*>(p);
    cc->print();
    cout << "-----" << endl;
    Parent& pp = dynamic_cast<Parent&>(c);
    pp.print();
    cout << "-----" << endl;
    Child& ccc = dynamic_cast<Child&>(pp);
    ccc.print();
    cout << "-----" << endl;
    //编译失败。转换后的类型必须为指针或引用
    //Parent ppp = dynamic_cast<Parent>(c);
    return 0;
}

结果:

结果

const_cast

  • const_cast用来将类型的const、volatile和__unaligned属性移除。常量指针被转换成非常量指针,并且仍然指向原来的对象;常量引用被转换成非常量引用,并且仍然引用原来的对象。
  • 不能直接对非指针和非引用的变量使用const_cast操作符去直接移除它的const、volatile和__unaligned属性。
int main()
{
    char buf[20] = "abcd";
    const char *p = buf;
    //编译报错,const修饰的内容无法修改
    //p[0] = '1';
    char *pp = const_cast<char *>(p);
    pp[0] = '1';
    cout << pp << endl;
    const char *str = "abcd";
    char *strp = const_cast<char *>(str);
    //这句运行出错,常量指针被转换成非常量指针,并且仍然指向原来的对象。而原来的"abcd"不可修改
    //*strp = '1';
    //不能直接对非指针和非引用的变量使用const_cast操作符去直接移除它的const、volatile和__unaligned属性。
    //int j = const_cast<int>(i);
    return 0;
}

reinterpret_cast

允许将任何指针类型转换为其它的指针类型;听起来很强大,但是也很不靠谱。它主要用于将一种数据类型从一种类型转换为另一种类型。它可以将一个指针转换成一个整数,也可以将一个整数转换成一个指针,在实际开发中,先把一个指针转换成一个整数,在把该整数转换成原类型的指针,还可以得到原来的指针值;特别是开辟了系统全局的内存空间,需要在多个应用程序之间使用时,需要彼此共享,传递这个内存空间的指针时,就可以将指针转换成整数值,得到以后,再将整数值转换成指针,进行对应的操作。

int main()
{
    char buf[20] = "abcd";
    int *p = reinterpret_cast<int *>(buf);
    char *pp = reinterpret_cast<char *>(p);
    cout << buf << endl << p << endl << pp << endl;;
    return 0;
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏黑泽君的专栏

c语言基础学习06_函数

============================================================================= 涉及...

2942
来自专栏blackheart的专栏

[C#1] 2-类型基础

1.System.Object CLR要求每个类型都要继承自System.Object[直接或者间接方式],如果不显示继承,编译器会自动为我们添加对System...

1967
来自专栏土豆专栏

Java面试之数据类型(一)

封装类是引用类型,基本类型在传递参数的时候都是按值传递,而封装类型是按引用传递的(其实引用也是按值传递的,但是传递的是对象的地址)

1722
来自专栏海天一树

小朋友学C++(12):多态

(一) 先编写函数: #include <iostream> using namespace std; class Shape { protected:...

3226
来自专栏python学习指南

Python迭代

本篇将介绍Python的迭代,更多内容请参考:Python学习指南 简介 在Python中,如果给定一个list或者tuple,我们可以通过for循环来遍...

2089
来自专栏java一日一条

Java中的静态绑定和动态绑定

一个Java程序的执行要经过编译和执行(解释)这两个步骤,同时Java又是面向对象的编程语言。当子类和父类存在同一个方法,子类重写了父类的 方法,程序在运行时调...

963
来自专栏racaljk

clang -O3 for循环的LLVM IR

这里删去了用处不大的内容,只保留了关键的LLVM IR。通过分析可以看到,如果循环小于8 LLVM IR会使用vector,vector使用SIMD指令高效进行...

1143
来自专栏java初学

final关键字

37212
来自专栏算法与数据结构

栈与递归 实现 十进制转二进制

6-4 十进制转换二进制(15 分) 本题要求实现一个函数,将正整数n转换为二进制后输出。 函数接口定义: void dectobin( int n ); 函数...

2955
来自专栏余林丰

5.比较排序之归并排序(非递归)

  在上一节中讲解了归并排序的递归版《4.比较排序之归并排序(递归)》,通常来讲,递归版的归并排序要更为常用,本节简单介绍下非递归版的归并排序。思路和递归版相...

2629

扫码关注云+社区

领取腾讯云代金券