char *p="1234567890",這樣的定義有問題嗎? 2008-08-25 19:47 348人閱讀 評論(1) 收藏 舉報
這個"1234567890"字串不再堆上也不在棧上!而是在文字常量區.而指標p在棧上
文字常量區:常量字串就是放在這裡的.程式結束後由系統釋放.
比如:
int a = 0; 全域初始化區
char *p1; 全域未初始化區
main()
{
int b; 棧
char s[] = "abc"; 棧
char *p2; 棧
char *p3 = "123456"; 123456/0在常量區,p3在棧上。
static int c =0; 全域(靜態)初始化區
p1 = (char *)malloc(10);
p2 = (char *)malloc(20);
分配得來得10和20位元組的地區就在堆區。
strcpy(p1, "123456"); 123456/0放在常量區,編譯器可能會將它與p3所指向的"123456"最佳化成一個地方。
}
一個由c/C++編譯的程式佔用的記憶體分為以下幾個部分
1、棧區(stack)— 由編譯器自動分配釋放 ,存放函數的參數值,局部變數的值等。其操作方式類似於資料結構中的棧。
2、堆區(heap) — 一般由程式員分配釋放, 若程式員不釋放,程式結束時可能由OS回收 。注意它與資料結構中的堆是兩回事,分配方式倒是類似於鏈表,呵呵。
3、全域區(靜態區)(static)—,全域變數和靜態變數的儲存是放在一塊的,初始化的全域變數和靜態變數在一塊地區, 未初始化的全域變數和未初始化的靜態變數在相鄰的另一塊地區。 - 程式結束後有系統釋放
4、文字常量區—常量字串就是放在這裡的。 程式結束後由系統釋放
5、程式碼區—存放函數體的二進位代碼。
char *p="1234567890";
這個語句是有問題的,解釋如下:
"1234567890"是一個字串常量,C和C++的標準都規定:任何試圖改變字串常量的行為其結果是“未定義的”。但是上述語句的指標定義卻不能阻止通過指標 p 來改變字串常量"1234567890"的這種行為,比如:*p = 'a';,編譯器發現不了這個錯誤。
正確的定義方式應該是:const char *p="1234567890";
那麼指標可以這樣初始化嗎?
回答又是可以的。由於曆史的原因,標準在這一點上作了一些“妥協”:既允許字元常量指標自動轉換為非常量指標。但是使用者應該知道這樣做存在著上述問題。因為這個原因,C++標準把這種轉換標記為“Deprecated”,應該避免使用。
而與此不同的是這種聲明方法:
char *p = (char *)10
這將使p指向地址0x0000000a,會個系統造成相當的破壞