为了克服C中强制类型转化带来的风险,C++引入了四种更加安全的强制类型转换运算符,分别是const_cast、static_cast、reinterpret_cast和dynamic_ccast,作用和区别如下:
const_cast
用法如下:
int n = 5;
int *k1 = const_cast<int*>(&n); //指针
*k1 = 1;
int &k2 = const_cast<int&>(n); //引用
k2 = 2;
const_cast去除const(volatile)属性,将只读变为可读写;
const_cast只针对指针、引用和this指针,其他情况的话就会出错,如下代码所示;
int n = 5;
int k1 = const_cast<int>(n); //编译出错
k1 = 1; //编译出错
static_cast
static_cast第一个作用是代替隐式转换 ,用于低风险的转化,即直接=就可以,编译器就不会报错的转换,比如:
int n = 10;
float f = 10.58;
char c = 'a';
void *p = nullptr;
int *k = static_cast<int*>(p);
n = static_cast<int>(c); //n就变成了97,ask码中字符a的位置
如果本身编译器就会报错的转换类型,比如将int装换成char,那stactic_cast也无法转换,编译失败
static_cast第二个作用是做基类与派生类的转换,派生类转换成子类是安全的,子类转换成派生类是不安全的,因为往往子类的内容比父类多,static_cast可以将子类转换成父类,但是不提供安全性检查
dynamic_cast
用于有虚函数的基类与其派生类之间的转换,特点是进行运行时检测,依赖于RTTI技术,但是有额外的函数开销,所以非必要的时候不使用。
为了弥补static_cast的第二个作用,所以希望有一种语法能检测基类和派生类之间转换的安全性,dynamic_cast就是来解决这个问题的(但是也有限制,只有包含虚函数的基类及其派生类可以使用而已),dynamic_cast编译时也不会出错,在运行时进行检测,如果转换失败返回nullptr(增加一个判断就可以检测是否转化出错)
reinterpret_cast
reinterpret代替显示转换,用于转换各种高风险的转换(隐式转换无法转换的),比如int转化成int,基类转化成子类等,但是不进行检查,只是进行强制的复制,有安全隐患,一般不用