1. RTTI
RTTI是 運行階段類型識別 (Runtime Type Identification)的簡稱。該特性是為了程序在運行階段確定對象類型提供一種標準方式。
1.1 RTTI的三個元素
由于只有包含虛函數的類層次結構,才能將派生類對象的地址賦給基類指針,因此RTTI只適用包含虛函數的類。
dynamic_cast
運算符將使用一個指向基類的指針來生成一個指向派生類的指針,否則,該運算符返回0—空指針。typeid
運算符返回一個指出對象的類型的值。type_info
結構存儲了有關特定類型的信息。
1.1.1 dynamic_cast運算符
dynamic_cast
運算符是RTTI中最常用的組件,其主要的功能是 確定是否可以安全地將對地址賦給特定類型的指針 ,如果可以則返回該特定類型的指針,如果不可以則返回0(空指針)。可以理解為dynamic_cast
可以用來類型轉換,如果可安全轉換,則轉換成相應的類型指針,否則直接返回0(空指針)。其語法如下:
dynamic_cast< type_name > (expression)
例如下面的例子:
class Base{
private:
double val;
public:
Base(double a = 0):val(a){}
virtial void show(){std::cout< "Base show!"< < std::endl();}
}
class BaseA:public: Base{
BaseA(double a = 0):Base(a){}
virtual void show(){std::cout < "BaseA show!"<
假設有下面的指針:
Base* bs = new Base;
Base* ba = new BaseA;
Base* bb = new BaseB;
對于下面類型的轉換:
BaseB* b1 = dynamic_cast
因此針對該例子,dynamic_cast
常用的場景如下:
auto bs = new BaseX;//BaseX為Base、BaseA、BaseB中的任意一種類型,假設其具體類型只有在程序運行過程中才能確定。
BaseA* pa;
if(pa = dynamic_cast
1.1.2 typeid運算符
typeid
運算符能夠確定兩個對象是否為同種類型,其可以接受兩種參數:
- 類名;
- 結果為對象的表達式。
其返回一個type_info
對象的引用,其實方法如下:
//需要包含頭文件#include< typeinfo >
A* a = new A;//A為一個類
if(typeid(A) == typeid(*a)){
std::cout < < "a的類型為A"<
針對上例,若a
是一個空指針,則typeid(*a)
會引發bad_typeid
異常,該異常類型是從exception
類派生出來,也是在typeinfo
中生聲明的。
1.1.3. type_info類
type_info
類主要存儲了有關特定類型的信息,其中包含了一個name()
成員,該成員函數主要用于調試,其需要與typeid
搭配使用,使用方法如下:
//class A;
A* a = new A;
std::cout< typeid(*a).name()<
2. 類型轉換運算符
2.1 dynamic_cast
前面已經介紹過其用法,該運算符的主要用途是,使得 能夠在類層次結構中進行向上轉換 (即派生類到基類的轉換,由于該轉換是is-a的關系,所以該轉換是安全的),而不允許其他的轉換。
2.2 const_cast
const_cast
運算符用于執行只有一種用途的類型轉化,即改變const
或volatile
,其語法與dynamic_cast
相同:
const_cast < type_name > (expression)
如果類型的其他方面也被修改,則上述類型轉換將出錯,也就是說除了const
或volatile
特征(有或無)可以不同外,type_name
和expression
的類型必須相同。假設A
和B
,則有下面的轉換:
A a;
const A* pa = &a;
A* pa1 = const_cast< A* >(pa); //正確
const B* pb = const_cast< const B* >(pa);//錯誤
提供該運算符的原因是,程序有時候可能需要一個這樣的值,它在大多數的時候是常量,而有時候又是可以修改的,在這種情況下,可以將這個值聲明為const
,并在需要修改它的時候,使用const_cast
。
2.3 static_cast
static_cast
運算符與其他類型轉換運算符的語法一樣:
static_cast < type_name > (expression)
static_cast
將expression
轉換為type_name
類型,主要用于非多態類型之間的轉換,不提供運行時的檢查來確保轉換的安全性。主要在以下幾種場合中使用:
- 用于類層次結構中 ,基類和子類之間指針和引用的轉換;當進行上行轉換(把派生類的指針或引用轉換成基類表示),這種轉換是安全的;當進行下行轉換(把基類的指針或引用轉換成派生類表示),這種轉換是不安全的,也需要程序員來保證;
- 用于基本數據類型之間的轉換 ,如把
int
轉換成char
,把int
轉換成enum
等等,這種轉換的安全性需要程序員來保證; - 把void指針轉換成目標類型的指針 ,是極其不安全的,也需要程序員來保證;
2.4 reinterpret_cast
reinterpret_cast
運算符與其他類型轉換運算符的語法一樣:
reinterpret_cast < type_name > (expression)
reinterpret_cast
運算符用于天生危險的類型轉換,因為其允許將任何指針類型轉換為其它的指針類型。reinterpret_cast
運算符并不會改變括號中運算對象的值,而是對該對象從位模式上進行重新解釋 。它主要用于將一種數據類型從一種類型轉換為另一種類型。
例如,它可以將一個指針轉換成一個整數,也可以將一個整數轉換成一個指針,然而,其并不支持所有類型的轉換,例如,可以將指針類型轉換為足以存儲指針表示的整形,但不能將指針轉換為更小的整形或浮點型。
-
轉換器
+關注
關注
27文章
8741瀏覽量
147715 -
存儲器
+關注
關注
38文章
7526瀏覽量
164162 -
運算符
+關注
關注
0文章
172瀏覽量
11095
發布評論請先 登錄
相關推薦
評論