C++ const修飾變量和修飾函數(shù)介紹
const修飾變量
關(guān)于const最常見的一個(gè)面試題是這樣的:char *const和const char*有什么區(qū)別,大家都知道const修飾符代表的是常量,即const修飾的變量一旦被初始化是不能被更改的,這兩個(gè)類型一個(gè)代表的是指針不可變,一個(gè)代表指針指向內(nèi)容不可變,但具體哪個(gè)對(duì)應(yīng)哪個(gè),很多人一直搞不清楚。
有這樣一個(gè)規(guī)律,const修飾的是它前面所有的數(shù)據(jù)類型,如果const在最前面,那么把它和它后面第一個(gè)數(shù)據(jù)類行交換.比如上面的const char*交換之后就是char const *,這樣一來就很清楚了,char *const p中的const修飾的是char *(注意,我們這里把char和*都算作一種類型,這時(shí)候const修飾的是char和*的組合,也就是字符串指針),是一個(gè)指針類型,所以這時(shí)候指針p是不能變的,比如下面這段代碼就會(huì)報(bào)錯(cuò)
char str1[]="str1";
char str2[]="str2";
char *const p = str1;
p = str2;
這時(shí)候p是一個(gè)指針常量,它是不能指向別的地方的,但是它本身指向的內(nèi)容是可以變的,比如下面的操作就是允許的
char str1[]="str1";
char *const p = str1;
p[0] = 'X';
printf("%s", str1);
這時(shí)候str1的值就變成了"Xtr1"
我們?cè)賮砜碿onst char *p,根據(jù)前面提到的規(guī)律,將const和它后面一個(gè)類型交換變成char const *p(其實(shí)這種寫法也是允許的,只是人們習(xí)慣將const寫在最前面),這時(shí)候const修飾的是char,也就是說p指向的字符內(nèi)容是不能變的。將上面兩個(gè)例子的char *const p全部改成const char *p,則結(jié)果正好相反,第一個(gè)可以編譯通過,第二個(gè)會(huì)報(bào)錯(cuò)。
其它時(shí)候就很好區(qū)分了,比如const int ,const string等等,總之,const修飾的是什么類型,這個(gè)類型的變量就不能被改變。
const修飾函數(shù)
先來看這樣一個(gè)函數(shù)
const char * func(const char *str) const;
這樣的函數(shù)比較夸張,有三個(gè)const,我們從左到右來一一說明:
1、第一個(gè)const修飾的是返回值,前面已經(jīng)說過,這里的const修飾的是char,也就是說返回值的內(nèi)容是不能被更改的
2、第二個(gè)const和第一個(gè)是一樣的,這種用的比較多,它作為函數(shù)參數(shù),表示的是這個(gè)參數(shù)在函數(shù)體內(nèi)是不能被改動(dòng)的(被傳進(jìn)來的實(shí)參并不要求是const類型),這樣做是為了防止函數(shù)對(duì)實(shí)參做一些意外的操作,你試想下,當(dāng)你調(diào)用一個(gè)函數(shù)時(shí),你傳進(jìn)去一個(gè)變量是"hello world!",調(diào)完函數(shù)之后變成了"fuck the world!",這實(shí)在是不可忍的,所以我們?cè)谠O(shè)計(jì)函數(shù)的時(shí)候,如果傳進(jìn)來的參數(shù)只作為讀取使用,最好是將參數(shù)設(shè)成const類型。很多公司在面試讓寫代碼的時(shí)候都會(huì)看中這個(gè)細(xì)節(jié),你注意了這個(gè)細(xì)節(jié)不一定說明你牛逼,但你若沒注意那肯定是會(huì)減分的。
3、再來說第三個(gè)const,按照我們最開始說的規(guī)律,const修飾的是它前面的所有數(shù)據(jù)類型,這里它前面的所有數(shù)據(jù)類型組合起來就是一個(gè)函數(shù),這種類型一般出現(xiàn)在類成員函數(shù)里,當(dāng)然,這里并不是說這個(gè)函數(shù)是不能變的,它代表的時(shí)這個(gè)函數(shù)不能改變類的成員變量,不管是public的還是private的
我們下面舉例主要說明第三種情況,來看這樣一個(gè)簡單的程序
#include<stdio.h>
class A
{
public:
A() : x(0), y(0){}
void func(const int p)
{
x = p;
y = p;
}
int getY()
{
return y;
}
int x;
private:
int y;
};
int main(int argc, char* argv[])
{
A a;
printf("x:%d y:%d\n", a.x, a.getY());
a.func(2);
printf("x:%d y:%d\n", a.x, a.getY());
return 0;
}
這段代碼是可以直接編譯過的,運(yùn)行結(jié)果是
x:0 y:0
x:2 y:2
我們稍作修改,將void func(const int p)改成void func(const int p) const再編譯,就會(huì)直接報(bào)錯(cuò),報(bào)錯(cuò)的兩行代碼是
x = p;
y = p;
也就是說const類型的函數(shù)試圖去修改類的成員變量是非法的,但是有一種情況例外,我們?cè)僭谏厦嫘薷牡幕A(chǔ)上做一點(diǎn)修改,將int x改成mutable int x,將int y改成mutable int y,這時(shí)候程序又可以正常運(yùn)行了,也就是說,如果成員變量是mutable類型的,它可以在任何場景下被修改。
相關(guān)文章
C語言數(shù)據(jù)結(jié)構(gòu)之線索二叉樹及其遍歷
這篇文章主要介紹了C語言數(shù)據(jù)結(jié)構(gòu)之線索二叉樹及其遍歷的相關(guān)資料,為了加快查找節(jié)點(diǎn)的前驅(qū)和后繼。對(duì)二叉樹的線索化就是對(duì)二叉樹進(jìn)行一次遍歷,在遍歷的過程中檢測節(jié)點(diǎn)的左右指針是否為空,如果是空,則將他們改為指向前驅(qū)和后繼節(jié)點(diǎn)的線索,需要的朋友可以參考下2017-08-08

C++17 使用 std::string_view避免字符串拷貝優(yōu)化程序性能

關(guān)于C/C++中typedef的定義與用法總結(jié)