溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何從C語(yǔ)言過(guò)渡到C++const

發(fā)布時(shí)間:2021-09-16 09:48:05 來(lái)源:億速云 閱讀:146 作者:柒染 欄目:編程語(yǔ)言

今天就跟大家聊聊有關(guān)如何從C語(yǔ)言過(guò)渡到C++const,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

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)寫(xiě)出下面這段代碼的執(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這句在編譯器看來(lái)應(yīng)該是

int a = 2 * 5 + 1;

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

#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常量用全大寫(xiě)字母命名。

特點(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)。如果想在其他文件中使用該如何寫(xiě)呢?

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

不過(guò)并不推薦這么使用,還是建議大家把const定義寫(xiě)在頭文件中,在需要的文件中包含這個(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ò)

這兩種寫(xiě)法都是非法的。

注意:依然可以用 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ò)。

這種寫(xiě)法的目的只是為了限制參數(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ì)把這種寫(xiě)法先過(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í),只能給相同情況的指針賦值

看完上述內(nèi)容,你們對(duì)如何從C語(yǔ)言過(guò)渡到C++const有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI