C++中强制类型转换
C++强制类型转换
C++中的强制类型转换虽然兼容C语言中的强制类型转换。但是并不建议在C++中使用C语言风格的强制类型转换。C++中的强制类型转换共有4个关键字分别是:static_cast,const_cast,reinterpret_cast,dynamic_cast.
static_cast
用于将一种数据类型转换成另一种数据类型,使用格式如下:
变量1 = static_cast<变量1数据类型>(另外一种数据类型变量或表达式);
例如:
int a = 1;
float b;
b = static_cast<float>(a); //类似于C语言中的b = (float)a;
const_cast
用于去除指针和引用的常量性,不能去除变量的常量性。使用格式如下:
指针或引用1 = const_cast<指针或引用1类型>(带常量性的指针或引用2);
例如:
#include <iostream>
using namespace std;
int main()
{
int a = 10;
const int *p = &a; // 被const修饰,不能使用该指针修改其指向内容的值
int *q;
q = const_cast<int *>(p); // 去除p的常量性给q,如果不去除直接赋值会报错
*q = 20;
cout<<"a的地址为:"<<&a<<" a的值为:"<<a<<endl;
cout<<"*q指向的地址为:"<<q<<" *q的值为:"<<*q<<endl;
}
执行结果如下:
a的地址为:0x61ff14a a的值为:20
*q指向的地址为:0x61ff14a *q的值为:20
为了加深理解,我们再看如下一段代码:
#include <iostream>
using namespace std;
const int & Add5(int &a);
int main()
{
const int a = 10; //a的值不允许修改
int b = 20;
const int *p = &a; //被const修饰,不能使用该指针修改其指向内容的值
int *q;
q = const_cast<int *>(p); //去除p的常量性给q,如果不去除直接赋值会报错
*q = 20;
cout<<"a的地址为:"<<&a<<" a的值为:"<<a<<endl;
cout<<"*q指向的地址为:"<<q<<" *q的值为:"<<*q<<endl;
cout<<"b的地址为:"<<&b<<" b的值为:"<<b<<endl;
int &m = const_cast<int &>(Add5(b)); //去掉函数的const属性,不去掉就赋值给不带const的引用会报错
cout<<"调用加5函数之后"<<endl;
cout<<"引用m指向的地址为:"<<&m<<" 引用m指向的值为:"<<m<<endl;
}
//返回输入值+5,返回的引用不可被修改
const int & Add5(int &a)
{
a+=5;
return a;
}
执行结果如下:
a的地址为0x61ff10 a的值为:10
*q指向的地址为:0x61ff10 *q的值为:20
b的地址为:0x61ff0c b的值为:20
调用加5函数之后
引用m指向的地址为:0x61ff0c 引用m指向的值为:25
上面这段程序有两点说明:
- 对变量a加上了const属性,因此a的值是不会被程序所改变的。即使我们将指向a的指针的const属性去掉重新赋值,也不会改变。但是会出现变量a和指针q指向地址相同,但值不同的现象。
- 对变量b的操作我们可以看出,const_cast不仅仅能去掉指针的const,也能去掉引用的const。不仅仅能去掉变量表达式的const也能去掉函数的const。
最后,在强调一遍。const_cast不能去掉变量的const.这是编程安全的基础。像下面这种操作是错误的。除了const之外,const_cast也能去掉volatile属性,和去掉const一样。
const int a = 10;
int b = const_cast<int>(a); //变量的const属性去不掉,错误
reinterpret_cast
reinterpret_cast这种强制类型转换是一种非常强的强制类型转换,它可以将任意两个无关的指针或引用进行转换。上面的static_cast进行强制类型转换时,会进行编译时的类型安全检查,即你可以将int转成float,将子类引用(指针)转成父类引用(指针),子转父和父转子都可以,但父转子不安全。但是你不能使用static_cast将两个无关的东西进行转换,比如两个无关的类,因为编译器在编译的时候会检查这个转换是否可行,很明显不可行。
ClassB *p = &b;
ClassA *q = static_cast<ClassA &>(p); //如果ClassA和ClassB有关系,编译能通过,否则不通过
但是如果使用的reinterpret_cast强制类型转换,就不会进行静态类型检查直接放行通过。当然如果这两个类如果没有关系,那么在运行的过程就会“跑偏出错”。为了防止在运行的过程中“跑偏”,我们除静态类型转换(static_cast)之外还需要一个动态的类型转换(dynamic_cast).
比较常用的一种reinterpret_cast的用法是,使用其将一个整形转化为一个IO设备的地址。比如我们写一段DSP程序,这个DSP外部数据总线上挂了一个FPGA,这段程序的作用就是读写FPGA中的的数据。下面给出了C风格的设备地址定义和C++风格的设备地址定义。
// C风格的设备地址定义
Uint16 *FpgaAddr0 = (Uint16*)0x200000; //Fpga第0个寄存器地址
//C++风格的设备地址定义
Uint16 *FpgaAddr0 = reinterpret_cast<Uint16 *>0x200000; //Fpga第0个寄存器地址
dynamic_cast
前面我们介绍过了static_cast,从名称上来看,两个一动(dynamic)一静(static)肯定有关系。我们知道static_cast会在编译的过程中进行安全性检查,而dynamic_cast会在运行的过程中进行安全性检查。这两个都带安全性检查可以防止错误的类型转换导致程序跑偏。
dynamic_cast既然是动态安全性检查,那么它肯定只能应用于指针或引用,不能用于内置的数据类型转换(内置的数据类型转换,在编译阶段由static_cast检查即可)。dynamic_cast不但检查两个指针是否属于同一个继承树(static_cast也检查这个),还会检测这种转换是否可行。如果可行就会返回一个新指针,并计算出为处理多继承的需要的必要的偏移量。如果不可行会返回NULL。因此即使我们使用reinterpret_cast强制类型转换骗过编译器编译成功,如果我们在使用前再使用dynamic_cast转换检测一下的话,还是能发现这个错误的转换的。
总结
- static_cast会进行静态的安全性检查,一般用于内置数据类型的转换和通常的类之间的转换。
- const_cast主要是用来去掉指针和引用的const和volatile类型。
- reinterpret_cast用于完全没有关系指针或引用之间的转换,比如字符指针转整形指针。
- dynamic_cast通常用于基类和派生类之间的相互转换。
C++中强制类型转换的更多相关文章
- java中强制类型转换
在Java中强制类型转换分为基本数据类型和引用数据类型两种,这里我们讨论的后者,也就是引用数据类型的强制类型转换. 在Java中由于继承和向上转型,子类可以非常自然地转换成父类,但是父类转换成子类则需 ...
- C语言中强制类型转换总结
C语言中强制类型转换总结 ● 字符型变量的值实质上是一个8位的整数值,因此取值范围一般是-128-127,char型变量也可以加修饰符unsigned,则unsigned char 型变量的取值范围 ...
- c#中的里氏转换和Java中强制类型转换在多态中的应用
在c#中: 注意: 子类并没有继承父类的构造函数,而是会默认调用父类那个无参数的构造函数. 如果一个子类继承了一个父类,那么这个子类除了可以使用自己的成员外,还可以使用从父类那里继承过来的成员.但是父 ...
- javascript中强制类型转换
javascript开发过程中,强制类型转换一般发生在条件判断和==运算符.其他情况,发生的类型转换(与这两种情况也是基本类似,属于万变不离其宗的范畴),暂不讨论. == 双等运算符 考虑代码: a ...
- C++解析(25):关于动态内存分配、虚函数和继承中强制类型转换的疑问
0.目录 1.动态内存分配 1.1 new和malloc的区别 1.2 delete和free的区别 2.虚函数 2.1 构造函数与析构函数是否可以成为虚函数? 2.2 构造函数与析构函数是否可以发生 ...
- 关于Java中强制类型转换的问题
为了更好的理解我们先看下面的例子: package com.yonyou.test; import java.util.ArrayList; import java.util.Iterator; im ...
- java中强制类型转换时,高位数截取成低位数的方法
/** * 强制类型转换中的补码.反码.原码一搞清楚 */ int b=233;//正整数强转 System.out.println((byte)b); //负数:原码的绝对值取反再加一 符号为不变 ...
- JS中强制类型转换
JavaScript提供了3种强制类型转换的方法 一.Boolean()方法 该方法将指定的参数转换成布尔型.Boolean(object).参数object可以是字符串对象.数值对象.DOM对象等. ...
- 【Java_基础】Java中强制类型转换
首先,狭义上的强制类型转换指的是引用类型,且是父类向子类转换,这种转换只牵扯到引用名义类型的转换,具体的对象内存没有发生一点变化. 而基本类型的转换与此不同,其数据确实发生了变化.如果是基本类型和其包 ...
随机推荐
- ArcGIS特殊标注效果的简单实现
1. 普通纯色背景:例如望仙亭,水垄沟: 方法: 2. 背景+边框 例如进入点 方法:
- 如何查看Python对象的属性
在Python语言中,有些库在使用时,在网络上找到的文档不全,这就需要查看相应的Python对象是否包含需要的函数或常量.下面介绍一下,如何查看Python对象中包含哪些属性,如成员函数.变量等,其中 ...
- C++ CSTRINGLIST用法
CStringList类成员 构造 CStringList 构造一个空的CString对象列表 首/尾访问 GetHead 返回此列表(不能是空的)中头部的元素 GetTail 返回此列表(不能是 ...
- python闭包的代码
- 用pandas读取excel报错
用pandas.read_execl()方法读取excel文件报错. 后来导入xlrd第三方库,就好了.
- CRM某些表加入审计
--更新参数文件,设置Audit等级 alter system set audit_trail=db,extended scope=spfile; --更新参数文件,开始Audit alter sys ...
- fiddler对浏览器、app抓包及证书安装(转)
http://blog.csdn.net/u011608531/article/details/50838227 1.fiddler对浏览器抓包 1.1 对浏览器的http的抓包 Capturing开 ...
- JMeter学习(十四)JMeter函数学习(转载)
转载自 http://www.cnblogs.com/yangxia-test JMeter函数是一些能够转化在测试树中取样器或者其他配置元件的域的特殊值.一个函数的调用就像这样:${_functio ...
- 使用Fiddler查看APP的请求接口、接口参数和返回值的方法
1.下载Fiddler,然后安装成功后. 2.开启代理的设置 3.查看电脑的ip, 4.建立一个wifi局域网,什么360wifi,猎豹wifi,腾讯wifi都可以,用安装手机接入到这个局域网的wif ...
- OpenSessionInViewFilter 的配置及作用(原文地址: http://blog.csdn.net/sunsea08/article/details/4545186)
spring为我们解决hibernate的Session的关闭与开启问题. Hibernate 允许对关联对象.属性进行延迟加载,但是必须保证延迟加载的操作限于同一个 Hibernate Sessio ...