字串常量放在記憶體中的靜態儲存區

2021-06-13 13:12:31 字數 2593 閱讀 3950

一、在c++中,記憶體分成5個區,他們分別是堆、棧、自由儲存區、全域性/靜態儲存區和常量儲存區。

棧,就是那些由編譯器在需要的時候分配,在不需要的時候自動清楚的變數的儲存區。裡面的變數通常是區域性變數、函式引數等。

堆:就是那些由new分配的記憶體塊,他們的釋放編譯器不去管,由我們的應用程式去控制,一般乙個new就要對應乙個delete。如果程式設計師沒有釋放掉,那麼在程式結束後,作業系統會自動**。

自由儲存區:就是那些由malloc等分配的記憶體塊,他和堆是十分相似的,不過它是用free來結束自己的生命的。(果真new 和malloc動態分配的記憶體不在乙個記憶體區域嗎?印象中好像是在一起!)

全域性/靜態儲存區:全域性變數和靜態變數被分配到同一塊記憶體中,在以前的c語言中,全域性變數又分為初始化的和未初始化的,在c++裡面沒有這個區分了,他們共同占用同一塊記憶體區。

常量儲存區:這是一塊比較特殊的儲存區,他們裡面存放的是常量,不允許修改(當然,你要通過非正當手段也可以修改,而且方法很多)

char *c="chenxi";

"chenxi"這個

字串常量

而且被放置在

記憶體的靜態儲存區(存放全域性變數和靜態變數)

。那一般的int i=1;

1也是常量,為什麼1就不被放置在此程式的記憶體靜態區了呢?

所有的字元竄常量都被放在靜態記憶體區

因為字串常量很少需要修改,放在靜態記憶體區會提高效率

先看乙個例子:

char str1 = "abc";

char str2 = "abc";

const char str3 = "abc";

const char str4 = "abc";

const char *str5 = "abc";

const char *str6 = "abc";

char *str7 = "abc";

char *str8 = "abc";

cout << ( str1 == str2 ) << endl;

cout << ( str3 == str4 ) << endl;

cout << ( str5 == str6 ) << endl;

cout << ( str7 == str8 ) << endl;

結果是:0 0 1 1

str1,str2,str3,str4是陣列變數,它們有各自的記憶體空間;

而str5,str6,str7,str8是指標,它們指向相同的常量區域。

解釋:

「abc」 存放於靜態儲存區(而不是堆和棧中)

char str1 = "abc"; //靜態儲存區中存放的「abc」

複製乙份副本給陣列str1,

放在棧中(也就是陣列str1中的位置),這時候「abc」在記憶體中有兩塊區域(分別在靜態儲存區和棧中)

const char *str5 = "abc"; //不複製副本,而是將指標str5指向靜態儲存區中存放「abc」的位址

問題的引入:

看看下面的程式的輸出:

#include

char *returnstr()  

int main()

這個沒有任何問題,因為"hello world!"是乙個字串常量,存放在靜態資料區,

把該字串常量存放的靜態資料區的首位址賦值給了指標,

所以returnstr函式退出時,該該字串常量所在記憶體不會被**,故能夠通過指標順利無誤的訪問。

但是,下面的就有問題:

#include

char *returnstr()

int main()

"hello world!"是乙個字串常量,存放在靜態資料區,沒錯,

但是把乙個字串常量賦值給了乙個區域性變數(char 型陣列),該區域性變數存放在棧中,

這樣就有兩塊內容一樣的記憶體,也就是說「char p="hello world!";」這條語句讓「hello world!」這個字串在記憶體中有兩份拷貝,乙份在動態分配的棧中,另乙份在靜態儲存區。這是與前者最本質的區別,

當returnstr函式退出時,棧要清空,區域性變數的記憶體也被清空了,

所以這時的函式返回的是乙個已被釋放的記憶體位址,所以列印出來的是亂碼。

如果函式的返回值非要是乙個區域性變數的位址,那麼該區域性變數一定要申明為static型別(在靜態儲存區)。如下:

#include

char *returnstr()

int main()

字串常量放在靜態儲存區

char c chenxi 書上說 chenxi 這個字串被當作常量而且被放置在此程式的記憶體靜態區。那一般的int i 1 1也是常量,為什麼1就不被放置在此程式的記憶體靜態區了呢?請高手指點!所有的字元竄常量都被放在靜態記憶體區 因為字串常量很少需要修改,放在靜態記憶體區會提高效率 還有,int...

關於字串常量在記憶體中的生命週期

字串char s hello 與char s hello 看似都是將hello字串的位址賦值給指標 p。但是前面乙個表示式是字串常量的位址賦值給指標 該指標指向的字串中的字元是不允許被更改的。而後面乙個表示式是將該字串的每乙個字元賦值給陣列,該指標指向的陣列的首位址,而陣列成員是變數,因此可以允許被...

關於字串常量在記憶體中的生命週期

字串char s hello 與char s hello 看似都是將hello字串的位址賦值給指標 p。但是前面乙個表示式是字串常量的位址賦值給指標 該指標指向的字串中的字元是不允許被更改的。而後面乙個表示式是將該字串的每乙個字元賦值給陣列,該指標指向的陣列的首位址,而陣列成員是變數,因此可以允許被...