成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

C++之RTTI機制

開發 前端
在C++中,只有類中包含了虛函數時才會啟用RTTI機制,也就是當存在多態時才會存在RTTI機制,因為不存在多態的話在編譯階段既可以確定類型信息。

RTTI簡介

RTTI(Runtime Type Indentification) 即運行階段類型識別。這是 C++新引進的特性之一。RTTI旨在為程序在運行階段確定對象的類型提供一種標準方式。

這RTTI聽起來是不是有點java中反射的味道?大差不差...

在C++中,只有類中包含了虛函數時才會啟用RTTI機制,也就是當存在多態時才會存在RTTI機制,因為不存在多態的話在編譯階段既可以確定類型信息。

運行時類型識別(RTTI)功能主要由以下兩個運算符實現:

  • typeid運算符,用于返回表達式的類型
  • dynamic_cast運算符,用于將基類的指針或引用安全地轉換成派生類的指針

RTTI與dynamic_cast

我們知道C++中的多態是基于虛函數的方式實現的,而含有虛函數的類都會有一個對應的虛函數表,而這個虛函數表會存有相關類型的type_info的地址, 因而可以說dynamic_cast為RTTI的一個應用。

因為dynamic_cast使用RTTI,所以它在轉換的過程中是可靠的,只有進行轉換的指針確實是指向指定的類型時才會轉換成功,否則就會轉換失敗,返回空指針。

例如以下的例子,第27行和第28行通過new不同的對象類型,會影響到第29行dynamic_cast的轉換結果:

#include <iostream>
class Base {
public:
    Base(){

    }
    virtual ~Base() {

    }
    virtual void f(){
        std::cout << "Base f" << std::endl;
    }
};

class Derived :public Base {
public:
    Derived(){

    }
    virtual ~Derived() {}
    void f() override{
        std::cout << "Derived f" << std::endl;
    }
};

int main() {
//    Base *base = new Base;
    Base *base = new Derived;
    Derived *derived = dynamic_cast<Derived*>(base);
    if(nullptr != derived){
        derived->f();
    } else{
        std::cout << "dynamic_cast null" << std::endl;
    }
    return 0;
}

RTTI與typeid

typeid當作用于指針時,返回的結果是該指針的靜態編譯時類型。typeid當作用于指針時,該指針必須是有效的,若是空指針,將返回bad_typeid異常。

typeid 運算符返回一個對type_info對象的引用,其中,type_info是在頭文件 typeinfo 中定義的一個類。type_info類重載了==和 != 運算符,以便可以使用這些運算符來對類型進行比較。

通過typeid 運算符我們就可以判斷一個指針指向的真實類似是否是派生類:

#include <iostream>
class Base {
public:
    Base(){

    }
    virtual ~Base() {

    }
    virtual void f(){
        std::cout << "Base f" << std::endl;
    }
};

class Derived :public Base {
public:
    Derived(){

    }
    virtual ~Derived() {}
    void f() override{
        std::cout << "Derived f" << std::endl;
    }
};

int main() {
    std::vector<Base*> vec;
    vec.emplace_back(new Base);
    vec.emplace_back(new Derived);
    for (auto base: vec) {
        std::cout << "clase Name:" << typeid(*base).name() << std::endl;
        if(typeid(*base) == typeid(Derived)){
            std::cout << "base的運行類型是Derived"  << std::endl;
        } else{
            std::cout << "base的運行類型是Base"  << std::endl;
        }
        // 調用f函數驗證一下上面的打印是否正確
        base->f();
    }
    return 0;
}

運行結果打印:

RTTI運行結果打印

需要注意的是以上實例代碼的第31和第32行,對于typeid運算符,typeid(*base)和typeid(base)它們得到的結果是不同的, typeid(*base)才能正確獲得指針的類型。因為*base是指針的真實數據內容,而base只是一個指針。

按照這個原理表達式typeid(base)獲得的類型永遠是Base的指針,即使指針base指向的可能是派生類Derived,但typeid(base)也無法獲得正確的類型, 務必要使用typeid(*base)。

責任編輯:趙寧寧 來源: 思想覺悟
相關推薦

2020-10-23 18:46:58

C++程序類別

2010-02-01 14:33:05

C++實現RTTI

2011-07-15 00:47:13

C++多態

2010-01-25 18:24:11

C++

2011-06-09 14:52:09

Pimpl機制

2011-06-09 15:04:22

RAII機制

2011-06-09 14:34:04

C++NVI

2011-07-14 17:45:06

CC++

2011-07-13 18:24:18

C++

2011-07-10 15:26:54

C++

2010-02-01 17:19:30

C++運行機制

2020-07-30 12:40:35

CC++編程語言

2010-02-04 11:23:25

C++反射機制

2010-02-02 15:30:05

C++ include

2023-11-28 11:51:01

C++函數

2023-12-13 10:51:49

C++函數模板編程

2024-02-01 00:10:21

C++PIMPL編程

2010-01-13 11:14:06

C++虛表

2010-02-06 14:04:58

C++內存使用機制

2011-07-15 01:38:56

C++this指針
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品久久久久永久免费观看 | 男人天堂手机在线视频 | 久久久久久久一区 | 免费黄视频网站 | 高清黄色网址 | 国产在线色 | 亚洲永久在线 | 能看的av网站 | 国产超碰人人爽人人做人人爱 | 免费观看的黄色网址 | 狠狠操狠狠干 | 久久精品色欧美aⅴ一区二区 | 久久高清精品 | 日韩视频免费看 | 国产日韩欧美激情 | 狠狠撸在线视频 | 欧美男人亚洲天堂 | 国产精品无码专区在线观看 | 91成人免费电影 | 91精品国产一区二区三区 | 成年视频在线观看 | 国产a区| 久久精品国产一区二区电影 | 国产伦精品一区二区三区精品视频 | 成人做爰9片免费看网站 | 国产精品毛片一区二区三区 | 欧美一区二区三区在线观看 | 国产在线观看一区二区 | 欧美一区视频 | 最近日韩中文字幕 | 欧美成人免费在线 | 国产一区二区精品在线 | 亚洲国产成人久久久 | 久久高清国产 | 亚洲视频免费在线观看 | 国产视频一区二区 | 欧美福利三区 | 久久亚洲精品国产精品紫薇 | 精品91av| 亚洲a一区二区 | 成人黄页在线观看 |