1. 程式人生 > >靈活而奇特的C++語言特性——const(一)

靈活而奇特的C++語言特性——const(一)

       學習了博主的《漫談繼承技術》系列博文之後,相信大家都有所收穫吧!這次博主將和大家一起探討 《靈活而奇特的C++語言特性》 ,主要包括引用、常量(const)、常量表達式(constexpr)、靜態(static)、外部(expert)、型別定義(typedef)、類型別名(aliases)、型別轉換、作用域解析、統一初始化、顯示轉換運算子、特性(attribute)、使用者自定義文字、標頭檔案、可變長度引數列表和前處理器巨集。儘管這個知識清單顯得有點凌亂,但是這些話題都是博主經過精心挑選,是容易混淆的語言特性。本篇我們來學習一下const,增進大家對《靈活而奇特的C++語言特性》的理解。

       const是constant的縮寫,指保持不變的量。編譯器會執行這一要求,任何嘗試改變常量的行為都會當作錯誤處理。此外,當啟用了優化時,編譯器可以利用此資訊生成更好的程式碼。關鍵字const主要有兩種相關的用法。可以用這個關鍵字標記變數或者引數,也可以用const標記方法。本篇博文主要探討const變數和引數的含義,希望對大家有一點幫助。

const變數和引數

在使用const之前,讓我們先來了解一下const的特性:

①const常量,在定義時必須初始化。

②const常量,在編譯的過程中將該常量替換為初始化時的值。

③可以通過強制型別轉換將常量的地址賦給指標變數,通過指標變數來修改所指向的空間裡的值。

咱們舉個栗子吧:

#include <iostream>

using namespacestd;

int main(intargc,char**argv)

{

    //定義const常量

    constintnCValue = 10;

    constdoublePI = 3.141592653;

    //constint size;

    //通過強制型別轉換,使指標指向const常量

    int*nPtr = (int*)&nCValue;

    double*dPtr = (double*)&PI;

    //通過指標修改const

修飾的常量值

    *nPtr =20;

    *dPtr =4.5;

    cout<< "*nPtr: "<< *nPtr << endl;

    cout<< "nCValue: "<< nCValue << endl;

    cout<< "*dPtr: "<< *dPtr << endl;

    cout<< "PI: " << PI << endl;

    return0;

}

程式執行結果:


         注意,上面的特性要視具體的編譯器而定,以上是我在VS 2013上的測試結果。我在圖釋中所說的整型包括bool、char、short、int和long型等資料型別。

         如果將上面程式碼中的“constint size;”解註釋,編譯器將會報以下錯誤:

 

       可能有的小夥伴會問:“C語言中的const和C++中的const有什麼不同,為什麼有的時候我將C++中編譯通過的程式碼放到C原始檔中就會出現編譯錯誤?”。善於發現問題,說明你已經比別人強很多。其實,C語言中const定義的是隻讀變數,不是真正意義上的常量,C++中const定義的才是真正意義上的常量。那什麼是真正意義上的常量呢?下面我們一起來揭祕一下吧。

#define _CRT_SECURE_NO_WARNINGS//關閉安全性檢測

#include <iostream>

using namespacestd;

enum COLOR{RED = 20, BLUE,GREEN };

int main(intargc,char**argv)

{

    constintsize = 20;

    intnArray[size] = { 0 };//C原始檔中次行語句報錯

    inti = 0;

    //初始化陣列

    for(auto&var : nArray)

    {

        var =++i;

    }

    //輸出陣列所有元素

    cout<< "nArray = {";

    foreach(autovar in nArray)

    {

        cout<< var << ", ";

    }

    cout<< "}" << endl;

    //switch的引數必須要是整型常量

    switch(size)//C中次行語句報錯

    {

    caseCOLOR::RED:

        printf("size is equal COLOR::RED\n");

        break;

    caseCOLOR::BLUE:

        printf("size is equal COLOR::RED\n");

        break;

    default:

        printf("size is not in COLOR\n");

        break;

    }

    return0;

}

程式執行結果:


       在C語言中,const修飾的是隻讀變數,其實它本質還是變數,在真正需要使用常量的地方都會報語法錯誤。而C++中對const進行了擴充,使其修飾的是真正意義上的常量。

       可以使用const來“保護”變數不被修改。這個關鍵字的一個重要用法是替換define來定義常量,這是const最直接的應用。上面程式中已經舉過栗子啦,這裡就不在贅述。可能有的小夥伴會問:“那const和define到底有什麼區別呢?還是隻是提供了一種新的定義常量的方式而已?”。當然不是啦,const比define更實用更安全,接下來就讓我們一起來探討const和define之間的區別吧。

const和define之間的主要區別:

         ①define定義的常量在預處理時進行替換,const定義的常量在編譯的過程中進行替換

         ②define只是進行簡單的字元替換,不進行型別檢查,const則進行型別檢查

         ③define定義的常量不會分配空間,而const則會分配空間

         ④define定義的常量不支援除錯(不能檢視它在記憶體空間裡的值),而const定義的常量則可以。

         ⑤define不能定義形參,const則可以定義形參

         關於這些區別的測試就留給大家啦,博主就不再編寫相關程式碼了。

       const修飾變數,這個變數包括指標變數。可能有的小夥伴聽說過指標常量和常量指標的概念,並對此十分困惑,總是搞混。沒關係,初學指標碰點壁也是很正常的。接下來我們就一起來聊聊它們之間的差異吧。

#include <iostream>

using namespacestd;

int main(intargc,char**argv)

{

    intnValue1 = 10;

    intnValue2 = 20;

    //nPtr是常量指標

    constint*nPtr = &nValue1;

    //nCPtr是指標常量

    int* const nCPtr = &nValue2;

    //常量指標:不能通過指標修改所指向物件的值,但可以更改其指向,指向其他物件

    //*nPtr= 30;

    nPtr =&nValue2;

    //指標常量:不能更改其指向,指向其他物件,但可以通過指標修改所指向物件的值

    //nCPtr= &nValue1;

    *nCPtr =50;

    cout<< "nValue1: "<< nValue1 << endl;

    cout<< "nValue2: "<< nValue2 << endl;

    return0;

}

程式執行結果:


       上述程式碼中“constint*nPtr = &nValue1;”也可以寫成“intconst *nPtr = &nValue1;”,其效果是一樣的。然而,這一規則卻不適應與“int* const nCPtr = &nValue2;”,將該語句寫為“int* nCPtr const = &nValue2;”或“intconst* nCPtr = &nValue2;”則不行,直接改變了語句,甚至無法通過編譯。

       可以嘗試著從右向左讀,“int* const nCPtr = &nValue2;”,const與nCPtr現結合,就知道nCPtr是一個指向int的const指標。另一方面,“int const*nPtr = &nValue1;”,“*”先和nPtr結合,就知道nPtr是一個指向const int的指標。

       const可以應用於引用,它通過比應用於指標更簡單。一是,引用預設為const,無法改變引用所指的物件。因此,無法顯式地將引用標記為const。二是,無法建立一個引用的引用,所以引用通過只有一層間接取值。獲取多層間接取值的唯一方法是建立指標的引用。舉個栗子:

constint* const * const* const * constPtr = nullptr;

constint* const * const* const * const&ref = Ptr;

       將物件作為引數傳遞時,預設選擇是const引用。只有在明確需要修改物件時,才能忽略const。const修飾函式引數的的場景我們已經見過,例如拷貝建構函式,建構函式引數等。這裡就不舉栗子了,如果你確實想看栗子的話,就去《靈活而奇特的C++語言特性——引用(下)》,那裡有一個封裝MyString類的栗子。

       關於const變數和引數的含義和使用我們就探討到這裡了,如果你還覺得意猶未盡,就請關注博主的《靈活而奇特的C++語言特性——const(二)》這篇博文,博主在那裡講述了const方法的含義和使用方法。

       如果想了解更多關於C++語言特性相關的知識,請關注博主《靈活而奇特的C++語言特性》系列博文,相信你能夠在那裡尋找到更多有助你快速成長和加深你對C++語言特性相關的知識和一些特性的理解和掌握。當然,如果你想了解關於繼承方面的技術,請關注博主《漫談繼承技術》系列博文。