标准C++中有四个类型转换符:static_cast、dynamic_cast、reinterpret_cast、和const_cast。下面介绍其中的两种。
第一种:static_cast
用法:static_cast < type-id > ( expression )
圆括号、尖括号都不能少,否则编译程序时会报错
功能:该运算符把expression转换为type-id类型,但没有运行时类型检查来保证转换的安全性。
①用于类层次结构中基类和子类之间指针或引用的转换。
进行上行转换(把子类的指针或引用转换成基类表示)是安全的;
进行下行转换(把基类指针或引用转换成子类表示)时,由于没有动态类型检查,所以是不安全的。
②用于基本数据类型之间的转换。如把int转换成char,把int转换成enum。这种转换的安全性也要开发人员来保证。
③把空指针转换成目标类型的空指针。
④把任何类型的表达式转换成void类型。
注意:static_cast不能转换掉expression的const、volitale、或者__unaligned属性。
第二种:dynamic_cast
主要用于执行“安全的向下转型(safe down casting)”,也就是说,要确定一个对象是否是一个继承体系中的一个特定类型。
用法:dynamic_cast < type-id > ( expression )
该运算符把expression转换成type-id类型的对象。Type-id必须是类的指针、类的引用或者void *;如果type-id是类指针类型,那么expression也必须是一个指针,如果type-id是一个引用,那么expression也必须是一个引用。
dynamic_cast主要用于类层次间的上行转换和下行转换,还可以用于类之间的交叉转换。
在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的;在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。
在类层次间进行上行转换时,dynamic_cast和static_cast的效果是一样的;在进行下行转换时,dynamic_cast具有类型检查的功能,比static_cast更安全。
#include <iostream>
using namespace std;
class B
{
public:
B(int n)
{
m_iNum=n;
}
virtual ~B(){}
virtual void foo(){};
private:
int m_iNum;
};
class D:public B
{
public:
D(int n):B(n)
{ m_szName = 20; }
~D(){}
void set(int n)
{
m_szName = n;
}
int get()
{
return m_szName ;
}
private:
int m_szName ;
};
void func(B *pb)
{
D *pd1 = static_cast<D *>(pb);
pd1->set(30);
cout<<"member of D: m_szName = "<<pd1->get()<<endl;
D *pd2 = dynamic_cast<D *>(pb);
if(pd2==NULL)
{
cout<<"pd2 is NULL"<<endl;
return ;
}
pd2->set(30);
cout<<"member of D: m_szName = "<<pd2->get()<<endl;
}
int main()
{
B b(10);
D d(10);
cout<<"pass on pointer of B:\n";
func(&b);
cout<<"pass on pointer of D:\n";
func(&d);
cout<<"\n";
return 0;
}
运行结果:
pass on pointer of B:
member of D: m_szName = 30
pd2 is NULL
pass on pointer of D:
member of D: m_szName = 30
member of D: m_szName = 30
另外,dynamic_cast还支持交叉转换。请看以下代码:
class A
{
public:
int m_iNum;
virtual void f(){}
};
class B:public A
{ };
class C:public A
{ };
void func()
{
B *p = new B;
p->m_iNum = 100;
C *pd1 = static_cast<C *>(p); //编译出错
C *pd2 = dynamic_cast<C *>(p); //pd2的结果为NULL
delete b;
}
另外两种类型转换的具体使用见:http://wenku.baidu.com/view/a9356b80d4d8d15abe234e32.html