介绍 C风格的强制类型转换(Type Cast)很简单,不管什么类型的转换统统是:TYPE b = (TYPE)a,但是c 风格的类型转换有不少的缺点,有的时候用c风格的转换是不合适的,因为它可以在任意类型之间转换,比如你可以把一个指向const对象的指针转换成指向非 const对象的指针,把一个指向基类对象的指针转换成指向一个派生类对象的指针,这两种转换之间的差别是巨大的,但是传统的c语言风格的类型转换没有区 分这些。还有一个缺点就是,c风格的转换不容易查找,他由一个括号加上一个标识符组成,而这样的东西在c++程序里一大堆。所以c++为了克服这些缺点,引进了4种类型转换操作符(C++风格的强制转换其他的好处是,它们能更清晰的表明它们要干什么。程序员只要扫一眼这样的代码,就能立即知道一个强制转换的目的。):
转换类型操作符 | 作用 |
---|---|
const_cast | 去掉类型的const或volatile属性 |
static_cast | 无条件转换,静态类型转换 |
dynamic_cast | 有条件转换,动态类型转换,运行时检查类型安全(转换失败返回NULL) |
reinterpret_cast | 仅重新解释类型,但没有进行二进制的转换 |
int c=static_cast(7.987);
如果涉及到类的话,static_cast只能在有相互联系的类型中进行相互转换,不一定包含虚函数。
class A
{};
class B:public A
{};
class C
{};
int main()
{
A* a=new A;
B* b;
C* c;
b=static_cast(a);
// 编译不会报错, B类继承A类
c=static_cast(a);
// 编译报错, C类与A类没有任何关系
return 1;
}
const_cast const_cast操作不能在不同的种类间转换。相反,它仅仅把一个它作用的表达式转换成常量。它可以使一个本来不是const类型的数据转换成const类型的,或者把const属性去掉。
int main() {struct T {
int i;
};
const T a;
//a.i = 10;
//直接修改const类型,编译错误
T &b = const_cast(a);
b.i = 10;
return 0;
}
dynamic_cast: 【C++|C++中四种类型转换方式__笔记】(1)其他三种都是编译时完成的,dynamic_cast是运行时处理的,运行时要进行类型检查。
(2)不能用于内置的基本数据类型的强制转换。
(3)dynamic_cast转换如果成功的话返回的是指向类的指针或引用,转换失败的话则会返回NULL。
(4)使用dynamic_cast进行转换的,基类中一定要有虚函数,否则编译不通过。
需要检测有虚函数的原因:类中存在虚函数,就说明它有想要让基类指针或引用指向派生类对象的情况,此时转换才有意义。
这是由于运行时类型检查需要运行时类型信息,而这个信息存储在类的虚函数表(关于虚函数表的概念,详细可见
只有定义了虚函数的类才有虚函数表。
(5) 在类的转换时,在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的。在进行下行转换 时,dynamic_cast具有类型检查的功能,比 static_cast更安全。向上转换即为指向子类对象的向下转换,即将父类指针转化子类指针。向下转换的成功与否还与将要转换的类型有关,即要转换的指针指向的对象的实际类型与转换以后的对象类型一定要相同,否则转换失败。
#include
#include
using namespace std;
class A
{
public:
virtual void f()
{
cout<<"hello"<(a1);
//结果为not null,向下转换成功,a1之前指向的就是B类型的对象,所以可以转换成B类型的指针。
if(b==NULL)
{
cout<<"null"<(a2);
//结果为null,向下转换失败
if(b==NULL)
{
cout<<"null"<(a);
//结果为null,向下转换失败
if(c==NULL)
{
cout<<"null"<
reinterpret_cast (interpret是解释的意思,reinterpret即为重新解释,此标识符的意思即为数据的二进制形式重新解释,但是不改变其值。)有着和C风格的强制转换同样的能力。它可以转化任何内置的数据类型为其他任何的数据类型,也可以转化任何指针类型为其他的类型。它甚至可以转化内置的数据类型为指针,无须考虑类型安全或者常量的情形。不到万不得已绝对不用。
仅重新解释类型,但没有进行二进制的转换:
1)转换的类型必须是一个指针,应用、算术类型、函数指针或者成员指针。
2)在比特级别上进行转换,可以把一个指针转换成一个整数,也可以把一个整数转换成一个指针(先把一个指针转换成一个整数,在把该整数转换成原类型的指针,还可以得到原先的指针值)。但不能将非32bit的实例转成指针。
3) 最普通的用途就是在函数指针类型之间进行转换。
4) 很难保证移植性。
int doSomething() {
return 0;
};
int main() {
typedef void(*FuncPtr)();
//FuncPtr is 一个指向函数的指针,该函数没有参数,返回值类型为 void
FuncPtr funcPtrArray[10];
//10个FuncPtrs指针的数组 让我们假设你希望(因为某些莫名其妙的原因)把一个指向下面函数的指针存入funcPtrArray数组:funcPtrArray[0] = &doSomething;
// 编译错误!类型不匹配,reinterpret_cast可以让编译器以你的方法去看待它们:funcPtrArray
funcPtrArray[0] = reinterpret_cast(&doSomething);
//不同函数指针类型之间进行转换return 0;
}
总结 去const属性用const_cast
基本类型转换用static_cast
多态类之间的类型转换用dynamic_cast
不同类型的指针类型转换用reinterpret_cast
推荐阅读
- #|c++中的四种类型转换
- C++学习笔记|C++11常用特性
- 2021SC@SDUSC|seccomp实现安全判题沙箱
- 单片机|SWM32系列教程3-时钟配置和GPIO
- 学习|20220609学习记录(st表、动态规划、数学)
- c++|c++运算符重载
- 算法|工程详细记录(超准确人脸检测(带关键点)YOLO5Face C++)
- c++|【华为外包面试】删除链表的倒数第 n 个结点,并且返回链表的头结点
- leetcode|LeetCode93