欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++中如何修改const變量你知道嗎

 更新時間:2022年03月08日 10:42:11   作者:heyabo  
這篇文章主要為大家詳細介紹了C++修改const變量,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助

一、結(jié)論

聲明:不同于C語言的const變量修改問題(可以通過指針間接修改const變量的值),這里只討論C++ 里的const。

C++ const 修飾符,表示常量,即如果以后保證不會修改則聲明為const,否則若要修改,那一開始為什么還要聲明為const呢?

根據(jù)C++標準,對于修改const變量,屬于:未定義行為(指行為不可預(yù)測的計算機代碼),這樣一來此行為取決于各種編譯器的具體實現(xiàn)(即不同編譯器可能表現(xiàn)不同)。

故結(jié)論就是:不建議這么做!

但是,是的,但是,網(wǎng)上論壇、博客里均有有關(guān)如何修改const變量的方法,其不是依賴于某種具體的編譯器,就是講的欠考慮。

方法是在定義變量的時候加上volatile關(guān)鍵字(沒有其他方法了嗎(比如,const_cast ...)? 是的,目前為止,我只知道這種方法是可能的):

const volatile int i = 10;

:關(guān)于volatile這里不細講,詳見:volatile 關(guān)鍵字。考慮到volatile的重要性,后面自己也會寫一篇關(guān)于volatile詳解的文章。

二、分析

為了說明問題,下面在三種編譯器環(huán)境下做幾個小實驗

1. g++ 

#include <stdio.h>
int main()
{
    const volatile int i = 10;
    int* pi = (int*)(&i);
    *pi = 100;
    printf("*pi: %d\n",*pi);
    printf("i: %d\n",i);
    printf("pi: %p\n",pi);
    printf("&i: %p\n", &i);
    return 0;
}

輸出結(jié)果:

gdb查看其匯編代碼命令:進入gdb,然后輸入:disass main):

可以看出:輸入*pi 和 i 時均是從堆棧(即內(nèi)存)中取數(shù)的。

反例:把 volatile關(guān)鍵字去掉:

#include <stdio.h>
int main()
{
    const int i = 10;
    int* pi = (int*)(&i);
    *pi = 100;
    printf("*pi: %d\n",*pi);
    printf("i: %d\n",i);
    printf("pi: %p\n",pi);
    printf("&i: %p\n", &i);
    return 0;
}

輸出結(jié)果:

由此可見:在沒有volatile關(guān)鍵字修飾時,const 變量 i 的值時沒有改變的。

運用gdb查看其匯編代碼:

注意此時(沒有加volatile修飾符),輸出 變量 i 的值時直接將 0xa(10)值(從符號表中取出的)輸出,即此處編譯器進行了優(yōu)化,沒有從內(nèi)存中讀。

注意到:指針 pi 和 &i(i 的地址)值卻是一樣的。So ,Why?

這就是C++中的常量折疊:指const變量(即常量)值放在編譯器的符號表中,計算時編譯器直接從表中取值,省去了訪問內(nèi)存的時間,從而達到了優(yōu)化。

而在此基礎(chǔ)上加上volatile修改符,即告訴編譯器該變量屬于易變的,不要對此句進行優(yōu)化,每次計算時要去內(nèi)存中取數(shù)。

這里也有個小細節(jié):每種編譯器對volatile修飾符的修飾作用效果不一致,有的就直接“不理會”,如VC++6.0編譯器(下面會講到)。

2. dev c++

運行結(jié)果與1(g++)一致。

3. VC++ 6.0

(1)添加volatile修飾符時,輸出結(jié)果(程序代碼同上):

 i 的值還是10,沒有改變!這是為什么呢?不急,先看下其匯編代碼:

注意:g++ 匯編代碼的mov指令 與 VC++ 6.0的mov指令不同(傳送方向相反)。

真相大白:雖然定義const變量的同時加上了volatile修飾符,但VC++ 6.0編譯器還是進行了優(yōu)化措施,輸出 i 時 從編譯器的符號表中取值,直接輸出。

(2)無 volatile 修飾符時。輸出結(jié)果:

i 的值沒有改變,預(yù)期中。其匯編代碼為:

結(jié)果與添加volatile時相同。

即在VC++6.0編譯環(huán)境下,在const變量定義時添加volatile修飾符,與不添加效果是一樣的。編譯器都采取了優(yōu)化(甚至把編譯器優(yōu)化選項關(guān)閉還是如此,有點恐怖...)。

4. VS 2010

再看下Microsoft編譯器家族的高級版本:

(1)添加 volatile 修飾符時,輸出結(jié)果:

 i 的值被成功修改了!

(2)無 volatile 修飾符時,輸出結(jié)果:

i 的值沒有被修改。

故:不建議修改const變量的值,即使修改也要熟悉當前使用的編譯器對于該 未定義行為 是如何解釋的。

總結(jié)

本篇文章就到這里了,希望能夠給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!  

相關(guān)文章

  • C語言qsort函數(shù)用冒泡排序?qū)崿F(xiàn)過程詳解

    C語言qsort函數(shù)用冒泡排序?qū)崿F(xiàn)過程詳解

    qsort函數(shù)是由C語言提供的標準庫函數(shù), 它的實現(xiàn)思想是快速排序。這篇文章主要介紹了C語言中qsort函數(shù)用法及用冒泡排序?qū)崿F(xiàn)qsort函數(shù)功能,需要的可以參考一下
    2023-02-02
  • C語言實現(xiàn)班級成績管理系統(tǒng)

    C語言實現(xiàn)班級成績管理系統(tǒng)

    這篇文章主要為大家詳細介紹了C語言實現(xiàn)班級成績管理系統(tǒng),文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-07-07
  • C++排序算法之插入排序

    C++排序算法之插入排序

    這篇文章主要為大家詳細介紹了C++排序算法之插入排序,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-08-08
  • C++實現(xiàn)漢諾塔算法經(jīng)典實例

    C++實現(xiàn)漢諾塔算法經(jīng)典實例

    這篇文章主要介紹了C++實現(xiàn)漢諾塔算法經(jīng)典實例,代碼簡潔高效,對于學習算法的朋友有一定的借鑒價值,需要的朋友可以參考下
    2014-07-07
  • C++的sstream標準庫詳細介紹

    C++的sstream標準庫詳細介紹

    以下是對C++中的的sstream標準庫進行了詳細的介紹,需要的朋友可以過來參考下
    2013-09-09
  • C語言實現(xiàn)個稅計算器

    C語言實現(xiàn)個稅計算器

    這篇文章主要為大家詳細介紹了C語言實現(xiàn)個稅計算器,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • C++實現(xiàn)假裝藍屏整蠱小程序

    C++實現(xiàn)假裝藍屏整蠱小程序

    因為工作的需要,有時候我們離開時并不想讓別人看到我們電腦的內(nèi)容,所以本文為大家分享了一個基于C++實現(xiàn)的假裝藍屏的小程序,需要的可以參考下
    2023-06-06
  • C++下標運算符[]重載代碼示例

    C++下標運算符[]重載代碼示例

    這篇文章主要給大家介紹了關(guān)于C++下標運算符[]重載的相關(guān)資料,C++ 規(guī)定下標運算符[]必須以成員函數(shù)的形式進行重載,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2024-01-01
  • C語言形參和實參傳值和傳址詳解刨析

    C語言形參和實參傳值和傳址詳解刨析

    形參出現(xiàn)在函數(shù)定義中,在整個函數(shù)體內(nèi)都可以使用, 離開該函數(shù)則不能使用。實參出現(xiàn)在主調(diào)函數(shù)中,進入被調(diào)函數(shù)后,實參變量也不能使用,形參和實參的功能是作數(shù)據(jù)傳送。發(fā)生函數(shù)調(diào)用時, 主調(diào)函數(shù)把實參的值傳送給被調(diào)函數(shù)的形參從而實現(xiàn)主調(diào)函數(shù)向被調(diào)函數(shù)的數(shù)據(jù)傳送
    2021-11-11
  • 樹形結(jié)構(gòu)的3中搜索方式示例分享

    樹形結(jié)構(gòu)的3中搜索方式示例分享

    樹的3中常見搜索方式,包括二叉樹方式(每一層只有0和1)、滿m叉樹(每一層都有0 到m - 1)、子集樹,也稱為全排列樹,需要的朋友可以參考下
    2014-02-02

最新評論