從C語言過渡到C++之const
1. 定義常量
1.1 C語言中定義常量的方法
在C語言從零開始這個(gè)系列中,我們講了C語言定義常量的方法。沒有看過的同學(xué)請(qǐng)參考:C語言從零開始(五)-常量&變量
為什么要定義常量我就不再贅述了,這里重點(diǎn)說說這么定義有什么不好。經(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)?define定義的常量是偽常量,它在參加編譯時(shí)做的是原樣字符替換。就是2 * SUM這句在編譯器看來應(yīng)該是
int a = 2 * 5 + 1;
如果你的本意是想得到12,那么定義中應(yīng)該這么寫:
#define SUM (5 + 1);
這樣的經(jīng)典錯(cuò)誤很多人都犯過,雖然道理大家都知道,但是總會(huì)因?yàn)榇中拇笠獾暨M(jìn)這個(gè)坑里。
于是,C++引入const常量徹底解決了這個(gè)問題。后來部分C語言的編譯器也開始支持const的使用,這就充分說明了它的價(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)前文件訪問。如果想在其他文件中使用該如何寫呢?
// file1 const int MAX = 10; // file 2 extern const int MAX;
不過并不推薦這么使用,還是建議大家把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)部的修改,如今越來越多的人喜歡這樣實(shí)現(xiàn):
void fun(int a) { const int& b = a; b++; // 非法操作 }
效果是完全一樣的。
3.2 const返回值
如果函數(shù)返回值是一個(gè)基本數(shù)據(jù)類型,用const修飾是沒有意義的。比如:
const int fun() { return 1; }
fun()函數(shù)的返回值是不可能做“左值”再被修改的,因?yàn)闆]人會(huì)這么使用:
fun() = 2;
編譯器也會(huì)把這種寫法先過濾掉。
一般,const只用來修飾返回值是一個(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ù)和返回值的情況。我們用一段代碼來看看容易出現(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部分中介紹的原則。在傳參和賦值的過程中需要注意:
指針內(nèi)容被const修飾時(shí),*p不可修改
指針內(nèi)容被const修飾時(shí),不能賦值給內(nèi)容非const的指針
指針變量和內(nèi)容都被const修飾時(shí),只能給相同情況的指針賦值
說起來有些拗口,仔細(xì)想想其實(shí)和第二部分所講的內(nèi)容相似。
OK,今天就先到這里。
上一篇:數(shù)據(jù)結(jié)構(gòu) 雙機(jī)調(diào)度問題的實(shí)例詳解
欄 目:C語言
下一篇:從C語言過渡到C++之基本變化
本文標(biāo)題:從C語言過渡到C++之const
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/1305.html
您可能感興趣的文章
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用函數(shù)刪除字符
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)式函數(shù)庫
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)數(shù)怎么表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段函數(shù)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排序法函數(shù)
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段函數(shù)
- 04-02C語言中怎么打出三角函數(shù) c語言中怎么打出三角函數(shù)的值
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求階乘


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達(dá)
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對(duì)數(shù)函數(shù)的表達(dá)式 c語言中對(duì)
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機(jī)閱讀
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 04-02jquery與jsp,用jquery
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置