C語言查漏補缺——const

一、const的作用
const修飾的變量,變成了一個不可修改的常量,比如
int c = 0;這個時候,c是可以修改的,但是加上const以後,c的值就變成常量,
const int c = 0; 如果這個時候令c = 2;那麼程序就會報錯。
這就是const的主要作用,但是,在C語言中,const修飾的變量真的就是一個常量嗎?

二、const的"bug"
我們看下面這段代碼

int main()
{
    const int cim_test = 1;

    printf("cim_test = %d\n", cim_test);

    int *p1 = &cim_test;

    *p1 = 3;
    printf("cim_test = %d\n", cim_test);

    return 0;
}
    這段代碼,運行結果如下

C語言查漏補缺——const
不是說好的const修飾的變量是一個常量嗎?常量是什麼,就是初始化以後,直到程序結束,值都不會改變的量。爲什麼這裏被改變了呢?

三、const的幾種情況
在回答上面的問題之前,我們先考慮const會出現的幾種情況,
1、const修飾static局部變量
2、const修飾全局變量
3、const修飾局部變量

第一種情況,這個時候的局部變量受到static的影響更大,因此,存儲在只讀存儲區上,無法進行修改。
第二種情況,這個時候的就要看編譯器了,如果是早期的編譯器,比如BCC這種的話,變量是存儲在全局數據區,而如果先一些現代c編譯器的話,那麼,編譯器會優化,將其存儲於只讀存儲區。
第三種情況,這是比較常見的,這種時候,變量是存儲在棧上的,因此,我們可以通過指針指向這段內存,然後修改這個值

因此,得出結論====>在C語言中,const修飾的變量,並不完全是常量。

下面是測試代碼

#include <stdio.h>

const int cig_test = 2;

int main()
{
    const int cim_test = 1;
    const static int cism_test = 3;

    printf("cim_test = %d\n", cim_test);

    int *p1 = &cim_test;

    *p1 = 3;
    printf("cim_test = %d\n", cim_test);

    int *p2 = &cig_test;
    *p2 = 4;       //不確定,需要知道編譯器是現代編譯器,還是古老的標誌編譯器
    printf("cig_test = %d\n", cig_test);

    int *p3 = &cism_test;
    *p3 = 5;      //報錯,加上static修飾的變量,存儲在只讀存儲區
    printf("cism_test = %d\n", cism_test);

    return 0;
}

四、const的擴展
我們可以利用const的這個屬性來規範一些代碼編寫。
1、const修飾的函數參數,在函數內部不允許修改。
2、const修飾的函數返回值,表示函數的返回值不可以修改
#include <stdio.h>

const char* f(const int i)  
{  
    i = 5;    //error,const變量不能作爲左值  

    return "Hello World";    //返回指向只讀存儲區的指針  
}  

int main()  
{  
    char* pc = f(0); //waring  

    printf("%s\n", pc);  

    pc[6] = '_';    //error,不能修改只讀存儲區的內容  

    printf("%s\n", pc);  

    return 0;  
}  
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章