C++ const修飾變量和修飾函數(shù)介紹
const修飾變量
關(guān)于const最常見(jiàn)的一個(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 *,這樣一來(lái)就很清楚了,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è)賮?lái)看const char *p,根據(jù)前面提到的規(guī)律,將const和它后面一個(gè)類型交換變成char const *p(其實(shí)這種寫法也是允許的,只是人們習(xí)慣將const寫在最前面),這時(shí)候const修飾的是char,也就是說(shuō)p指向的字符內(nèi)容是不能變的。將上面兩個(gè)例子的char *const p全部改成const char *p,則結(jié)果正好相反,第一個(gè)可以編譯通過(guò),第二個(gè)會(huì)報(bào)錯(cuò)。
其它時(shí)候就很好區(qū)分了,比如const int ,const string等等,總之,const修飾的是什么類型,這個(gè)類型的變量就不能被改變。
const修飾函數(shù)
先來(lái)看這樣一個(gè)函數(shù)
const char * func(const char *str) const;
這樣的函數(shù)比較夸張,有三個(gè)const,我們從左到右來(lái)一一說(shuō)明:
1、第一個(gè)const修飾的是返回值,前面已經(jīng)說(shuō)過(guò),這里的const修飾的是char,也就是說(shuō)返回值的內(nèi)容是不能被更改的
2、第二個(gè)const和第一個(gè)是一樣的,這種用的比較多,它作為函數(shù)參數(shù),表示的是這個(gè)參數(shù)在函數(shù)體內(nèi)是不能被改動(dòng)的(被傳進(jìn)來(lái)的實(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)來(lái)的參數(shù)只作為讀取使用,最好是將參數(shù)設(shè)成const類型。很多公司在面試讓寫代碼的時(shí)候都會(huì)看中這個(gè)細(xì)節(jié),你注意了這個(gè)細(xì)節(jié)不一定說(shuō)明你牛逼,但你若沒(méi)注意那肯定是會(huì)減分的。
3、再來(lái)說(shuō)第三個(gè)const,按照我們最開(kāi)始說(shuō)的規(guī)律,const修飾的是它前面的所有數(shù)據(jù)類型,這里它前面的所有數(shù)據(jù)類型組合起來(lái)就是一個(gè)函數(shù),這種類型一般出現(xiàn)在類成員函數(shù)里,當(dāng)然,這里并不是說(shuō)這個(gè)函數(shù)是不能變的,它代表的時(shí)這個(gè)函數(shù)不能改變類的成員變量,不管是public的還是private的
我們下面舉例主要說(shuō)明第三種情況,來(lái)看這樣一個(gè)簡(jiǎn)單的程序
#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;
}
這段代碼是可以直接編譯過(guò)的,運(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;
也就是說(shuō)const類型的函數(shù)試圖去修改類的成員變量是非法的,但是有一種情況例外,我們?cè)僭谏厦嫘薷牡幕A(chǔ)上做一點(diǎn)修改,將int x改成mutable int x,將int y改成mutable int y,這時(shí)候程序又可以正常運(yùn)行了,也就是說(shuō),如果成員變量是mutable類型的,它可以在任何場(chǎng)景下被修改。
相關(guān)文章
C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之線索二叉樹(shù)及其遍歷
這篇文章主要介紹了C語(yǔ)言數(shù)據(jù)結(jié)構(gòu)之線索二叉樹(shù)及其遍歷的相關(guān)資料,為了加快查找節(jié)點(diǎn)的前驅(qū)和后繼。對(duì)二叉樹(shù)的線索化就是對(duì)二叉樹(shù)進(jìn)行一次遍歷,在遍歷的過(guò)程中檢測(cè)節(jié)點(diǎn)的左右指針是否為空,如果是空,則將他們改為指向前驅(qū)和后繼節(jié)點(diǎn)的線索,需要的朋友可以參考下2017-08-08C++中形參和實(shí)參的區(qū)別及說(shuō)明
這篇文章主要介紹了C++中形參和實(shí)參的區(qū)別及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-02-02

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

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