C語(yǔ)言到C++(2) - const

C++中最早引入const是為了替代#define,后來(lái)又衍生出了其它用法徘六。這一篇中我們來(lái)詳細(xì)介紹const的各種常見(jiàn)用法锄贼。

1. 定義常量

1.1 C語(yǔ)言中定義常量的方法

C語(yǔ)言從零開(kāi)始這個(gè)系列中,我們講了C語(yǔ)言定義常量的方法注簿。沒(méi)有看過(guò)的同學(xué)請(qǐng)參考:C語(yǔ)言從零開(kāi)始(五)-常量&變量

為什么要定義常量我就不再贅述了契吉,這里重點(diǎn)說(shuō)說(shuō)這么定義有什么不好。經(jīng)常有這樣的面試題:請(qǐng)寫出下面這段代碼的執(zhí)行結(jié)果:

#include <stdio.h>

#define SUM 5 + 1;

void main()
{
    int a = 2 * SUM;
    printf("%d", a);
}

經(jīng)常有人答12诡渴,其實(shí)結(jié)果是11捐晶。不信你用計(jì)算機(jī)運(yùn)行一下試試。

為什么會(huì)錯(cuò)呢妄辩,因?yàn)?code>#define定義的常量是偽常量惑灵,它在參加編譯時(shí)做的是原樣字符替換。就是2 * SUM這句在編譯器看來(lái)應(yīng)該是

int a = 2 * 5 + 1眼耀;

如果你的本意是想得到12英支,那么定義中應(yīng)該這么寫:

#define SUM (5 + 1);

這樣的經(jīng)典錯(cuò)誤很多人都犯過(guò),雖然道理大家都知道哮伟,但是總會(huì)因?yàn)榇中拇笠獾暨M(jìn)這個(gè)坑里干花。

于是妄帘,C++引入const常量徹底解決了這個(gè)問(wèn)題。后來(lái)部分C語(yǔ)言的編譯器也開(kāi)始支持const的使用把敢,這就充分說(shuō)明了它的價(jià)值寄摆。

1.2 const常量

在C++中,我們用下面的形式定義常量:

const int MONTH = 12;
const int SUM = 5 + 1;

嚴(yán)格意義上講修赞,const常量應(yīng)該叫做“常變量”婶恼,它定義了一個(gè)值不會(huì)被修改的變量。

為了代碼風(fēng)格統(tǒng)一柏副,我們依然習(xí)慣把const常量用全大寫字母命名勾邦。

特點(diǎn)

const常量與普通常量最大的不同有兩點(diǎn):

  • 值不能改變
  • 可以用作數(shù)組大小的定義

例如:

const int MAX = 10;
int arr[MAX] = {0};
for (int i = 0; i < MAX; i++)
{
    // Do something
}

1.3 作用范圍

const定義的常量的作用域類似與static,只能被當(dāng)前文件訪問(wèn)割择。如果想在其他文件中使用該如何寫呢眷篇?

// file1
const int MAX = 10;
// file 2
extern const int MAX;

不過(guò)并不推薦這么使用,還是建議大家把const定義寫在頭文件中荔泳,在需要的文件中包含這個(gè)頭文件蕉饼。

2. 指針與const

const的修飾特點(diǎn)是修飾離它最近的部分。它一般有兩種用法玛歌。

2.1 指向const變量的指針

讓指針指向一個(gè)const對(duì)象昧港,防止指針修改所指向的值。

int age = 30;
const int* ptr = &age;

這段代碼定義了一個(gè)指針ptr支子,它指向一個(gè)const int類型的數(shù)據(jù)创肥,不可修改。

    *ptr += 1;  // 報(bào)錯(cuò)
    cin >> *ptr;    // 報(bào)錯(cuò)

這兩種寫法都是非法的值朋。

注意:依然可以用 age變量修改叹侄。

2.2 const指針

將指針本身聲明為一個(gè)常量,防止指針位置改變

int a = 3;
int* const p = &a;

p++; // 錯(cuò)誤

注意:只有const指針能夠指向const變量昨登,例如:

const int a = 9;
const int* p = &a;  // 正確
int* ptr = &a;      //錯(cuò)誤

特殊使用:

const int* const p = &a;

這句話的意思是指針變量和指向的地址中的內(nèi)容都不可變

3. 函數(shù)與const

3.1 const參數(shù)

如果希望參數(shù)在函數(shù)內(nèi)部不被修改趾代,可以用const修飾,如下:

void fun(const int a)
{
    a++; // 非法操作
}

由于a被const修飾為常變量丰辣,因此再對(duì)它進(jìn)行a++操作就會(huì)報(bào)錯(cuò)稽坤。

這種寫法的目的只是為了限制參數(shù)在函數(shù)內(nèi)部的修改,如今越來(lái)越多的人喜歡這樣實(shí)現(xiàn):

void fun(int a)
{
    const int& b = a;
    b++; // 非法操作 
}

效果是完全一樣的糯俗。

3.2 const返回值

如果函數(shù)返回值是一個(gè)基本數(shù)據(jù)類型尿褪,用const修飾是沒(méi)有意義的。比如:

const int fun()
{
    return 1;
}

fun()函數(shù)的返回值是不可能做“左值”再被修改的得湘,因?yàn)闆](méi)人會(huì)這么使用:

fun() = 2;

編譯器也會(huì)把這種寫法先過(guò)濾掉杖玲。

一般,const只用來(lái)修飾返回值是一個(gè)類的對(duì)象的函數(shù)淘正。例如:

class A
{
public:
    A()
    {
        m_i = 0;
    }

    A(int i) : m_i(i){}

    void Modify(int i)
    {
        m_i = i;
    }

private:
    int m_i;
};

A GetA()
{
    return A(1);
}

const A GetConstsA()
{
    return A(1);
}

void Update(A& a)
{
    a.Modify(2);
}

void Update2(const A& a)
{
    A m = a;
    m.Modify(2);
}

int main()
{
    GetA() = A(1);      // 正確
    GetA().Modify(5);   // 正確

    GetConstsA() = A(1);    // 報(bào)錯(cuò)
    GetConstsA().Modify();  // 報(bào)錯(cuò)

    Update(GetA());         // 正確
    Update(GetConstsA());   // 報(bào)錯(cuò)
    Update2(GetConstsA());  // 正確

    return 0;
}

能看懂其中的奧秘嗎摆马?總結(jié)一下臼闻,const修飾的返回值如果是類的對(duì)象,那么:

  • 這個(gè)返回值不能做左值(放在等號(hào)左邊被賦值或者調(diào)用其成員函數(shù))
  • 這個(gè)返回值的別名必須也被const修飾

4. 舉一反三

知道了一般參數(shù)和返回值被const修飾的情況囤采,我們應(yīng)該能夠推導(dǎo)出const修飾指針參數(shù)和返回值的情況述呐。我們用一段代碼來(lái)看看容易出現(xiàn)的錯(cuò)誤。

void fun1(int* p)
{
    // Do nothing
}

void fun2(const int* cp)
{
    *cp = 3; // 錯(cuò)誤
    int i = *cp; 
    int* ip2 = cp; // 錯(cuò)誤
}

const char* fun3()
{
    return "result of function fun3()";
}

const int* const fun4()
{
    static int i;
    return &i;
}

int main()
{
    int x = 0;
    int* p = &x;
    const int* cp = &x;

    fun1(p);
    fun1(cp); // 錯(cuò)誤

    fun2(p);
    fun2(cp);

    char* cp = fun3();  // 錯(cuò)誤
    const char* ccp = fun3();

    int* p2 = fun4(); // 錯(cuò)誤
    const int* const ccp = fun4();
    const int* cp2 = fun4();
    *fun4() = 1; // 錯(cuò)誤

    return 0;
}

這段程序的各種賦值其實(shí)完全符合第2部分中介紹的原則蕉毯。在傳參和賦值的過(guò)程中需要注意:

  • 指針內(nèi)容被const修飾時(shí)乓搬,*p不可修改
  • 指針內(nèi)容被const修飾時(shí),不能賦值給內(nèi)容非const的指針
  • 指針變量和內(nèi)容都被const修飾時(shí)代虾,只能給相同情況的指針賦值

說(shuō)起來(lái)有些拗口进肯,仔細(xì)想想其實(shí)和第二部分所講的內(nèi)容相似。

OK棉磨,今天就先到這里江掩。

我是天花板,讓我們一起在軟件開(kāi)發(fā)中自我迭代乘瓤。
如有任何問(wèn)題环形,歡迎與我聯(lián)系。


上一篇:C語(yǔ)言到C++(1) - 基本變化

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末衙傀,一起剝皮案震驚了整個(gè)濱河市斟赚,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌差油,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,542評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件任洞,死亡現(xiàn)場(chǎng)離奇詭異蓄喇,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)交掏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,596評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門妆偏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人盅弛,你說(shuō)我怎么就攤上這事钱骂。” “怎么了挪鹏?”我有些...
    開(kāi)封第一講書人閱讀 158,021評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵见秽,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我讨盒,道長(zhǎng)解取,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 56,682評(píng)論 1 284
  • 正文 為了忘掉前任返顺,我火速辦了婚禮禀苦,結(jié)果婚禮上蔓肯,老公的妹妹穿的比我還像新娘。我一直安慰自己振乏,他們只是感情好蔗包,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,792評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著慧邮,像睡著了一般调限。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上赋咽,一...
    開(kāi)封第一講書人閱讀 49,985評(píng)論 1 291
  • 那天旧噪,我揣著相機(jī)與錄音,去河邊找鬼脓匿。 笑死淘钟,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的陪毡。 我是一名探鬼主播米母,決...
    沈念sama閱讀 39,107評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼毡琉!你這毒婦竟也來(lái)了铁瞒?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 37,845評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤桅滋,失蹤者是張志新(化名)和其女友劉穎慧耍,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體丐谋,經(jīng)...
    沈念sama閱讀 44,299評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡芍碧,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,612評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了号俐。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片泌豆。...
    茶點(diǎn)故事閱讀 38,747評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖吏饿,靈堂內(nèi)的尸體忽然破棺而出踪危,到底是詐尸還是另有隱情,我是刑警寧澤猪落,帶...
    沈念sama閱讀 34,441評(píng)論 4 333
  • 正文 年R本政府宣布贞远,位于F島的核電站,受9級(jí)特大地震影響笨忌,放射性物質(zhì)發(fā)生泄漏兴革。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,072評(píng)論 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望杂曲。 院中可真熱鬧庶艾,春花似錦、人聲如沸擎勘。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 30,828評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)棚饵。三九已至煤裙,卻和暖如春艘策,著一層夾襖步出監(jiān)牢的瞬間啥么,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,069評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工蜡娶, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留欣硼,地道東北人题翰。 一個(gè)月前我還...
    沈念sama閱讀 46,545評(píng)論 2 362
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像诈胜,于是被迫代替她去往敵國(guó)和親豹障。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,658評(píng)論 2 350

推薦閱讀更多精彩內(nèi)容