C++进阶——类型转换符(参考百度文库)

       标准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

           在上面的代码段中,在函数func的参数列表中pb为基类对象指针,可接收B类型对象指针,也可以接收D类型对象指针。如果pb指向一个D类型的对象,pd1和pd2是最终结果一样的,并且对这两个指针执行D类型的任何操作都是安全的;但是,如果pb指向的是一个 B类型的对象,那么pd1将是一个指向B类型对象的指针,对它进行D类型的操作将是不安全的(如访问m_szName),而pd2将是一个空指针。另外要注意:B要有虚函数,否则会编译出错;static_cast则没有这个限制。这是由于运行时类型检查需要运行时类型信息,而这个信息存储在类的虚函数表(关于虚函数表的概念,详细可见<Inside c++ object model>)中,只有定义了虚函数的类才有虚函数表,没有定义虚函数的类是没有虚函数表的。

        另外,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

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值