日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網為廣大站長提供免費收錄網站服務,提交前請做好本站友鏈:【 網站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747

C++四種強制類型轉換介紹

 

接上文:C語言的隱式類型轉換和顯示類型轉換

隱式類型轉換是編譯器自動隱式進行的,需要在代碼中體現,而顯示類型轉換由程序員明確指定。

C++支持C風格的強制轉換,但是C風格的強制轉換可能帶來一些隱患,讓一些問題難以發現。

所以C++提供了一組適用于不同場景的強制轉換的函數:

  • static_cast
  • dynamic_cast
  • const_cast
  • reinterpret_cast

下面對這四種轉換操作的適用場景分別進行說明。

1、static_cast

static_cast<type>(expression)

該運算符把 expression 轉換為 type 類型,主要用于基本數據類型之間的轉換,如把 uint 轉換為 int,把 int 轉換為 double 等。

uint x = 1;
int y = static_cast<int>(x); // 轉換正確
int x = 1;
double y = static_cast<double>(x); // 轉換正確

需要注意的是:static_cast 沒有運行時類型檢查來保證轉換的安全性,需要程序員來判斷轉換是否安全。

int x = -1;
uint y = static_cast<uint>(x) // 轉換錯誤
double x = 1.23;
int y = static_cast<int>(x) // 轉換丟失精度

static_cast 還可用于類層次結構中,基類和派生類之間指針或引用的轉換,但也要注意:

  • static_cast 進行上行轉換是安全的,即把派生類的指針轉換為基類的;
  • static_cast 進行下行轉換是不安全的,即把基類的指針轉換為派生類的。
// 上行轉換,派生類→基類
Derive* d = new Derive();
Base* b = static_cast<Base*>(d);// 下行轉換,基類→派生類
Base* b = new Base();
Derive* d = static_cast<Derive*>(b);

這是因為派生類包含基類信息,所以上行轉換(只能調用基類的方法和成員變量),一般是安全的;

而基類沒有派生類的任何信息,而下行轉換后會用到派生類的方法和成員變量,這些基類都沒有,很容易“指鹿為馬”,或指向不存在的空間。

2、dynamic_cast

dynamic_cast<type>(expression)

dynamic_cast 主要用于類層次間的上行轉換或下行轉換。在進行上行轉換時,dynamic_cast 和 static_cast 的效果是一樣的,但在下行轉換時,dynamic_cast 具有類型檢查的功能,比 static_cast 更安全。

比如下面這段代碼:

#include <IOStream>
using namespace std;
class Base {
public:
    virtual void Say() {
        cout << "I am Base." << endl;
    }};class Derive : public Base {
public:
    virtual void Say() {
        cout << "I am Derive." << endl;
    }};int main(){    // 上行轉換
    Derive* d1 = new Derive();
    cout << "d1: " << d1 << endl;
    Base* b1 = dynamic_cast<Base*>(d1);
    cout << "b1: " << b1 << endl;
  
    // 下行轉換
    Base* b2 = new Base();
    cout << "b2: " << b2 << endl;
    Derive* d2 = dynamic_cast<Derive*>(b2);
    cout << "d2: " << d2 << endl;
    return 0;
}

運行結果為:

C++四種強制類型轉換介紹

 

在進行下行轉換時,從基類 b2 到 d2 時,d2 會改為空指針(0x0),這正是 dynamic_cast 提升安全的功能。這個檢查主要來自虛函數表。

在C++面向對象的思想中,虛函數是實現多態的關鍵機制。當一個類中有虛函數時,那么編譯器就會構建出一個虛函數表來指示這些函數的地址。當用基類的指針指向派生類的對象,調用方法時就會根據虛函數表找到對應派生類的方法。

注意:B 要有虛函數,否則會編譯出錯;static_cast則沒有這個限制。

這是由于運行時類型檢查需要運行時類型信息,而這個信息存儲在類的虛函數表,只有定義了虛函數的類才有虛函數表,沒有定義虛函數的類是沒有虛函數表的。

 

3、const_cast

const_cast<type>(expression)

該運算符用來修改 expression 的 const 或 volatile 屬性。這里需要注意:expression 和 type 的類型一樣的。

比如下面的代碼,指針 px 由于有 const 修飾,無法直接通過其修改 x 的值,但又期望能修改 x 的值時,怎么辦呢?這時就需要用到 const_cast。

int main()
{    int x = 1;
    cout << "before: " << x << endl;
      const int* px = &x;
    // *px = 2; // 編譯錯誤
      int* py = const_cast<int*>(px);
    *py = 2;
      cout << "px: " << px << endl;
    cout << "py: " << py << endl;
    cout << "after : " << x << endl;
      return 0;
}

運行結果為:

C++四種強制類型轉換介紹

 

可以看出,px 和 py 指向同一個地址,但通過 py 就可以修改 x 的值了。

這是因為通過const_cast,就把 const 類型的指針 px 轉換成非 const 類型的指針 py 了。

4、reinterpret_cast

reinterpret_cast<type>(expression)

該運算符可以把一個指針轉換成一個整數,也可以把一個整數轉換成一個指針。

int main()
{    int* p = new int(5);
    uint64_t p_val = reinterpret_cast<uint64_t>(p);
      cout << "p    :" << p << endl;
    cout << "p_val:" << hex << p_val << endl;
      return 0;
}

上述代碼把指針 p 的地址值轉換成了 uint64_t 類型的整數值,運行結果為:

C++四種強制類型轉換介紹

 

這個轉換是“最不安全”的。不推薦使用。

綜上,在使用強制類型轉換時,需要首先考慮清楚使用目的,總結如下:

  • static_cast:基本類型轉換,低風險;
  • dynamic_cast:類層次間的上行轉換或下行轉換,低風險;
  • const_cast:去 const 屬性,低風險;
  • reinterpret_cast:轉換不相關的類型,高風險。

分享到:
標簽:轉換 類型
用戶無頭像

網友整理

注冊時間:

網站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網站吧!
最新入駐小程序

數獨大挑戰2018-06-03

數獨一種數學游戲,玩家需要根據9

答題星2018-06-03

您可以通過答題星輕松地創建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學四六

運動步數有氧達人2018-06-03

記錄運動步數,積累氧氣值。還可偷

每日養生app2018-06-03

每日養生,天天健康

體育訓練成績評定2018-06-03

通用課目體育訓練成績評定