温故而知新 C++ 类型转换

C语言类型转换

在C语言里用到的类型转换方式,一般都是用强制类型转换,语法:(类型说明符)(表达式),例如: (float)a 把a转换为实型,(int)(x+y) 把x+y的结果转换为整型。
C语言这种赋值时的类型转换形式可能会使人感到不精密和不严格,因为不管表达式的值怎样,系统都自动将其转为赋值运算符左部变量的类型。

C++类型转换

const_cast,字面上理解就是去const属性。
static_cast,命名上理解是静态类型转换。如int转换成char。
dynamic_cast,命名上理解是动态类型转换。如子类和父类之间的多态类型转换。
reinterpret_cast,仅仅重新解释类型,但没有进行二进制的转换。

const_cast:

该运算符用来修改类型的const或volatile属性。除了const 或volatile修饰之外, type_id和expression的类型是一样的。

#include <stdio.h>

using namespace std;

struct Data{
    int value;
};

int main(int argc, _TCHAR* argv[])
{
    const Data data1 = {10};
    //data1.value = 15; //error C3892: “data1”: 不能给常量赋值
    Data &data2 = const_cast<Data &>(data1);
    data2.value = 20;
    printf("data1.value = %d data2.value = %d\n",data1.value,data2.value);

    const int a = 10;
    int *b = const_cast<int*>(&a);
    *b = 20;
    printf("a = %d b == &a is %d\n",a,b == &a);

    system("pause");

    return 0;
}

dynamic_cast
有条件转换,动态类型转换,运行时类型安全检查(转换失败返回NULL):
1. 安全的基类和子类之间转换。
2. 必须要有虚函数。
3. 相同基类不同子类之间的交叉转换。但结果是NULL。

class BaseClass {
public:
int m_iNum;
virtualvoid foo(){}; //基类必须有虚函数。保持多台特性才能使用dynamic_cast
};

class DerivedClass: public BaseClass {
public:
char*m_szName[100];
void bar(){};
};

BaseClass* pb =new DerivedClass();
DerivedClass *pd1 = static_cast<DerivedClass *>(pb); //子类->父类,静态类型转换,正确但不推荐
DerivedClass *pd2 = dynamic_cast<DerivedClass *>(pb); //子类->父类,动态类型转换,正确

BaseClass* pb2 =new BaseClass();
DerivedClass *pd21 = static_cast<DerivedClass *>(pb2); //父类->子类,静态类型转换,危险!访问子类m_szName成员越界
DerivedClass *pd22 = dynamic_cast<DerivedClass *>(pb2); //父类->子类,动态类型转换,安全的。结果是NULL

static_cast:

①用于类层次结构中基类(父类)和派生类(子类)之间指针或引用的转换。
进行上行转换(把派生类的指针或引用转换成基类表示)是安全的;
进行下行转换(把基类指针或引用转换成派生类表示)时,由于没有动态类型检查,所以是不安全的。
②用于基本数据类型之间的转换,如把int转换成char,把int转换成enum。这种转换的安全性也要开发人员来保证。
③把空指针转换成目标类型的空指针。(int* 转 long* error C2440: “static_cast”: 无法从“int *”转换为“long *”)
④把任何类型的表达式转换成void类型。

reinterpret_cast:

reinterpret_cast<type-id> (expression)
type-id 必须是一个指针、引用、算术类型、函数指针或者成员指针。它可以把一个指针转换成一个整数,也可以把一个整数转换成一个指针(先把一个指针转换成一个整数,再把该整数转换成原类型的指针,还可以得到原先的指针值)

主要是将原地址的值重新强制定义为新类型。

reinterpret_cast与static_cast的类型转换比较:

#include <stdio.h>

using namespace std;

class A
{
public:
    int m_a;
};

class B
{
public:
    int m_b;
};

class C:public A,public B
{

};

int main(int argc, _TCHAR* argv[])
{

    C* c = new C();
    c->m_a = 1;
    c->m_b = 2;

    A* a1 = static_cast<A*>(c);
    A* a2 = reinterpret_cast<A*>(c);

    B* b1 = static_cast<B*>(c);
    B* b2 = reinterpret_cast<B*>(c);

    printf("c = %p,a1 = %p,b1 = %p\n",c,a1,b1);
    printf("c = %p,a2 = %p,b2 = %p\n",c,a2,b2);

    printf("a1.m_a = %d, b1.m_b = %d\n",a1->m_a,b1->m_b);
    printf("a2.m_a = %d, b2.m_b = %d\n",a2->m_a,b2->m_b);

    system("pause");
    return 0;
}

运行结果:

c = 00392950,a1 = 00392950,b1 = 00392954
c = 00392950,a2 = 00392950,b2 = 00392950
a1.m_a = 1, b1.m_b = 2
a2.m_a = 1, b2.m_b = 1

由运行结果可以很明显地看到,使用reinterpret_cast进行强制转换的时候b2直接是从c的地址开始,将c转换成了b2,而使用static_cast的时候,b1是进行了一个sizeof(A)的一个偏移量的位移,也就是说在多重继承的情况下,将子类的对象指针转换成基类对象指针,使用static_cast才是安全的。reinterpret_cast的目的是将类型重定义,除非你非常明确地希望指向的那一块内存是希望定义到你要转换的目标,否则慎用。

reinterpret_cast的一个使用例子:

#include <stdio.h>

using namespace std;

typedef void(*FuncPtr)();
typedef int(*FuncPtr2)();

void fun1()
{
}

int fun2()
{
    return 10;
}

int main(int argc, _TCHAR* argv[])
{
    FuncPtr fun[10];
    fun[0] = &fun1;

    //fun[1] = &fun2; //error C2440: “=”: 无法从“int (__cdecl *)(void)”转换为“FuncPtr”
    //fun[1] = static_cast<FuncPtr>(&fun2); //error C2440: “static_cast”: 无法从“int (__cdecl *)(int)”转换为“FuncPtr”

    fun[1] = reinterpret_cast<FuncPtr>(&fun2); //用强制转换将int fun 转为 void fun 保存起来

    FuncPtr2* pfun = reinterpret_cast<FuncPtr2*>(&fun[1]); //在需要使用的时候,再转换回来

    printf("run pfun = %d\n",(*pfun)()); //打印得到 "run pfun =10"

    system("pause");
    return 0;
}
时间: 2024-10-27 18:19:07

温故而知新 C++ 类型转换的相关文章

More Effective C++

条款一:指针与引用的区别 指针与引用看上去完全不同(指针用操作符'*'和'->',引用使用操作符'.'),但是它们似乎有相同的功能.指针与引用都是让你间接引用其他对象.你如何决定在什么时候使用指针,在什么时候使用引用呢? 首先,要认识到在任何情况下都不能用指向空值的引用.一个引用必须总是指向某些对象.因此如果你使用一个变量并让它指向一个对象,但是该变量在某些时候也可能不指向任何对象,这时你应该把变量声明为指针,因为这样你可以赋空值给该变量.相反,如果变量肯定指向一个对象,例如你的设计不允许变量为

javascript 基础教程[温故而知新一]

子曰:“温故而知新,可以为师矣.”孔子说:“温习旧知识从而得知新的理解与体会,凭借这一点就可以成为老师了.“ 尤其是咱们搞程序的人,不管是不是全栈工程师,都是集十八般武艺于一身.不过有时候有些知识如果有很久没用了的话,就会忘记,甚至是忘的你一点都想不起来,尤其是一些基础的东西.所以我才打算写个"温故而知新"的系列博文出来,一来是这些基础的东西我比较健忘,以后方便自己翻阅:二来是希望可以帮助到一些刚入门的朋友.这个系列记录的所有知识点都是最最最(重要的事情说三遍)基础的知识.大部分都是我

Javascript类型转换的规则实例解析

http://www.jb51.net/article/79916.htm 类型转换可以分为隐式转换和显式转换,所谓隐式转换即程序在运行时进行的自动转换,显式转换则是人为的对类型进行强制转换.Javascript的变量是松散类型的,它可以存储Javascript支持的任何数据类型,其变量的类型可以在运行时被动态改变.请看示 例: ? 1 2 3 var n = 10; n = "hello CSSer!"; n = {}; 上面的示例中,首先声明n变量并初始化其值为10(整数类型),接

进制、类型转换、提升

进制 java支持四种进制表示 进制 示例 前缀 备注 十进制 int x = 10;     二进制 int x = 0b10; 0b或0B 计算机储存 八进制 int x = 010; 0 兼容以前程序 十六进制 int x = 0xceaf 0x或0X 0~9,a~f 细节: 一个十六进制位可以表示四个二进制位 当数值位数较多时,可以使用下划线对数字进行隔开,例:1_230_456 字面常量 整数:字面常量为int型,如果要表示long类型的字面常量,使用L或l做后缀,建议使用L,因为字母

第二节 变量 、 基本类型 、 运算符 、 表达式 、 数据 、 类型转换 、 常量

一:变量 定义:在程序运行中随时可以发生变化的量 变量声明:变量的声明只能以数字,字母,下划线,且字母不能开头,同时不能是C#中的关键字. 变量使用:变量必须先声明才能使用,变量赋值必须是对应类型,int age=true:这样就会报错. 二:常量 定义:在程序的生存期内不发生更改的不可变值 常量声明:const关键字  public const int age=10; age在程序内不会变,值一直是10: 三:内置数据类型(数值类型和非数值类型) 数值类型:分为整型和非整型 整型如下图: 非整

03 php 数据类型:整数,进制转换,浮点,字符,布尔,数组,空类型,类型转换,算术运算,比较运算

03 数据类型:整数,进制转换,浮点,字符,布尔,数组,空类型,类型转换, 算术运算,比较运算,逻辑运算,短路现象, 三目运算符,字符型运算: 数据类型 整体划分 标量类型: int, float, string, bool 复合类型: array,     object 特殊类型: null,     resouce 整数类型int, integer 3种整数表示法 十进制写法:123: $n1 = 123; 八进制写法: 0123 $n2 = 0123; 十六进制写法: 0x123 $n3

java 复习 - 数据类型转换

基本数据类型转换两天原则 低精度的值可以直接赋值给高精度的变量,直接变成了高精度 高精度的值不可以直接赋值给低精度的变量. 不同类型变量混合运算之后,得到的结果是精度最高的类型 基本数据类型和字符串之间的转换 基本数据类型转换成字符串 使用String 的valueOf()函数. 使用方法: String.valueOf(各种基本类型)  得到一个字符串.例子如下:  package typeConvertTest; public class TypeConvertTest{  public s

C++:类型转换

5.3 类型转换 5.3.1 系统预定义类型间的转换 1. 隐式转换: 例如: int x=5,y;             y=3.5+x; //系统会自动先将int型的5转换为double型的5.0,在和3.5进行加法运算. 隐式类型转换的主要规则: a.在赋值表达式A=B的情况下,赋值运算符右端B的值需要转换为A的类型后进行赋值. b.当char或short类型变量与int类型变量进行运算时,将char或short类型转换为int类型. c.当两个操作对象类型不一致时,在算术运算前,级别低

基本数据类型转换

自动类型转换:容量小的数据类型自动转换成容量大的数据类型.            char                           int long float double byte     short 一字节是8位,一字是2字节. 有多种类型的数据计算时.系统会将所有的数据类型转换成最大的数据然后进行计算. byte short char之间不会自动转换,他们三者计算会首先转换成int在进行计算. 当把任何基本类型的数据和自负类型的数据进行链接运算(+),得到的是字符类型的数据.