C++中 Const修飾的量叫常量,與普通變數的區別是什麼? C++中有兩點1:編譯方式不同,在編譯的過程中,所有在程式中出現名字的地方都會被初值替換掉 2:不能作為左值了 const 修飾的量常出現的使用錯誤有: 1:常量作為左值修改值 (因為這種方式可以直接修改常量的值) 2:把常量的地址泄漏給 ...
C++中
Const修飾的量叫常量,與普通變數的區別是什麼?
C++中有兩點1:編譯方式不同,在編譯的過程中,所有在程式中出現名字的地方都會被初值替換掉 2:不能作為左值了
const 修飾的量常出現的使用錯誤有:
1:常量作為左值修改值 (因為這種方式可以直接修改常量的值)
2:把常量的地址泄漏給一個普通的指針或普通的引用變數 (因為這種方式可以間接修改常量的值如下示例1代碼)
示例1:
const int a=10;
int p =&a; //p=20; 編譯報錯 不能將 const int*(整型常量的地址) 賦值給 int *(普通的int指針)
const 和一級指針結合方式有如下種類
C++的語言規範: const修飾的是離他最近的類型
const int *p;
p指針指向的內容不能再被修改,但是p可以指向任意的int類型的記憶體
int const *p;
這種方式同上
int * const p;
const 修飾的類型是 int *,修飾的表達式是p, 即 p的值(也是地址)不能被修改了,即p不能再指向其他記憶體地址了,但是p所指向的那個記憶體的值得內容可以修改 即 *p=200;
cont int * const p;
const p 是指 p不能再指向其他記憶體地址了, const int * 是指 p指向的內容不能在修改了.
示例2:如下
const int b=10;
const int * p=&b;(OK 編譯通過)
int *pp=&b (NOT OK, 相當於將 const int * 轉為 int *)
int c=10;
int *p1=&c; (OK)
const int *p2=&c; *(OK 相當於將 int * 轉為 const int )
int * const p3=&c; (OK)
int * p=nullptr;
int * const p1=nullptr;
cout<<typeid(p)<<endl; // 結果是 int *
cout<<typeid(p1)<<endl;// 結果是 int *
**結論:如果const 右側沒有指針即沒有*的話,那const不參與類型, 即 int * const p1, 這個const 不參與類型, p1的類型仍然是 int * 見下麵代碼
所以示例3:
int c=10;
int * const p3 = &c; (OK, int * 轉為 int * const 根據上面的結論 const 不參與類型, 其實還是 int * 轉為 int *)
示例4:
int x=100;
const int *p=&x;
int *p1=p; // 錯誤 相當於 const int * 轉為 int * 是不可以的
const和二級指針的結合
const 與二級指針的結合分成以下幾種
const int ** p
根據C++編程規範,const 修飾離他最近的類型, const 修飾的類型是int ,修飾的表達式是 **p , 是指 **p的值不能被修改
int * const * p
const 修飾的類型是 int*, 修飾的表達式是 *p , 指 *p 的值不能被修改, p的值可以修改 **p 的值也可以修改
int ** const p;
const 修飾的類型是 int** , 修飾的表達式是p ,指p的值不能被修改, *p的值可以修改,**p的值也可以修改
看以下示例是否能通過編譯
示例5:
int a =10;
int *p =&a;
const int ** q =&p; (這句話無法通過編譯)
上面一行代碼相當於 int ** 轉成 const int **
怎麼理解呢?
如果
int a=10;
int *p=&a;
const int ** q =&p;
上面的代碼可以編譯通過
首先 q 和p指向的是同一塊記憶體地址
如果有
const int b=200;
q =&b;
上面這兩行代碼,b是const int常量, 現在我們將b的地址放入q中,即q=&b,
由於*q和p是同一塊記憶體地址,那麼我們是不是就把一個常量的地址即b的地址,就暴露給了一個普通指針p呢?
這將會造成通過普通指針p很容易修改常量b的值了,所以const int ** q = &p; 是不可以的。那麼怎麼解決呢?
第一種辦法:
直接把int *p 加上const不就可以了嗎?不就限制了p修改常量的值了嗎?
第二種辦法:
int a=10;
int *p=&a
const int * const *q=&p;
我就不允許 *q =&b, 不就堵住了把一個常量的地址放入普通指針的口了嗎?
示例6:
int a =10;
const int *p =&a;
int ** q =&p; (這句話無法通過編譯)
上面一行代碼相當於 const int ** 轉成 int ** 更加不可以了
如何理解?
const int *p =&a 相當於一級指針P已經做承諾,我指向的內容是不可變的,通過p,是不允許修改指向值得內容的
那麼 int ** q =&p ,如果可以通過編譯的話,那麼我可以 **q 去修改a的值了,那不是讓 一級指針的const修飾了個寂寞嗎?
同理
int **q = int * const *p 可以轉嗎?
int * const *p const 修飾的類型是int *,修飾的表達式是 *p,是指 p所指向的內容是不允許修改的,
如果上面的轉換可以,那麼 **q 是不是可以輕易修改 最底層的值了呢?打破了上面的 的限制,所以結論是不可以的
int *const *q =int ** 可以轉嗎?
int * const *q 只是表達了 *q 的值是不能修改了而且,所以可以轉