在前面不知一次說過,定義變量時一定要初始化,尤其是數組和結構體這種占用內存大的數據結構。在使用數組的時候經常因為沒有初始化而產生“燙燙燙燙”這樣的野值,俗稱“亂碼”。
每種類型的變量都有各自的初始化方法,memset()函數可以說是初始化內存的“萬能函數”,通常為新申請的內存進行初始化工作。它是直接操作內存空間,mem即“內存(memory)”的的意思。該函數的原型為:
1 #include<string.h> 2 void *memset(void *s,int c,unsigned long n);
函數的功能是:將指針變量s所指向的前n字節的內存單元用一個“整數”c替換,注意c是int型。s是void*型的指針變量,所以它可以為任何類型的數據進行初始化。
memset()的作用是在一段內存塊中填充某個給定的值。因為它只能填充一個值,所以該函數的初始化為原始初始化,無法將變量初始化為程序中需要的數據。用memset初始化完后,后面程序中再向該內存空間中存放需要的數據。
memset一般使用“0”初始化內存單元,而且通常是給數組或結構體進行初始化。一般的變量如char、int、float、double等類型的變量直接初始化即可,沒有必要使用使用memset。如果用memset的話反而顯得麻煩。
當然,數組也可以直接進行初始化,但memset是對較大的數組或結構體進行清零初始化的最快方法,因為它是直接對內存進行操作的。
這時有人會問:“字符串數組不是最好用'\0'進行初始化嗎?那么可以用 memset 給字符串數組進行初始化嗎?也就是說參數 c 可以賦值為'\0'嗎?”
可以的。雖然參數 c 要求是一個整數,但是整型和字符型是互通的。但是賦值為 '\0' 和 0 是等價的,因為字符 '\0' 在內存中就是 0。所以在 memset 中初始化為 0 也具有結束標志符 '\0' 的作用,所以通常我們就寫“0”。
memset 函數的第三個參數 n 的值一般用 sizeof() 獲取,這樣比較專業。注意,如果是對指針變量所指向的內存單元進行清零初始化,那么一定要先對這個指針變量進行初始化,即一定要先讓它指向某個有效的地址。而且用memset給指針變量如p所指向的內存單元進行初始化時,n 千萬別寫成 sizeof(p),這是新手經常會犯的錯誤。因為 p 是指針變量,不管 p 指向什么類型的變量,sizeof(p) 的值都是 4。
1 #include<stdio.h> 2 #include<string.h> 3 int main(void) 4 { 5 int i;//循環變量 6 char str[10]; 7 char *p=str; 8 memset(str,0,sizeof(str));//只能寫sizeof(str),不能寫sizeof(p) 9 for (i=0;i<10;i++) 10 { 11 printf("%d\x20",str[i]); 12 } 13 printf("\n"); 14 return 0; 15 }
根據memset函數的不同,輸出結果也不同,分為以下幾種情況:
memset(p, 0, sizeof(p)); //地址的大小都是4字節 0 0 0 0 -52 -52 -52 -52 -52 -52
memset(p, 0, sizeof(*p)); //*p表示的是一個字符變量, 只有一字節 0 -52 -52 -52 -52 -52 -52 -52 -52 -52
memset(p, 0, sizeof(str));0 0 0 0 0 0 0 0 0 0
memset(str, 0, sizeof(str)); 0 0 0 0 0 0 0 0 0 0
memset(p, 0, 10); //直接寫10也行, 但不專業 0 0 0 0 0 0 0 0 0 0
