memset使用注意

2021-07-27 10:44:14 字數 3375 閱讀 9215

原文:

memest原型(please type "man memset" in your shell)

void *memset(void *s,

int c, size_t n);

memset:作用是在一段記憶體塊中填充某個給定的值,它對較大的結構體或陣列進行清零操作的一種最快方法。

常見的三種錯誤

第一: 搞反了c 和 n的位置.

一定要記住 如果要把乙個char a[20]清零, 一定是 memset(a, 0, 20)

而不是 memset(a, 20,

0)第二: 過度使用memset, 我想這些程式設計師可能有某種心理陰影, 他們懼怕未經初始化的記憶體, 所以他們會寫出這樣的**:

char buffer[20];

memset(buffer, 0, sizeof((char)*20));

strcpy(buffer, "123");

這裡的memset是多餘的. 因為這塊記憶體馬上就被覆蓋了, 清零沒有意義.

第三: 其實這個錯誤嚴格來講不能算用錯memset, 但是它經常在使用memset的場合出現

int some_func(struct something *a)

:為何要用memset置零?memset( &address, 0, sizeof(address));經常看到這樣的用法,其實不用的話,分配資料的時候,剩餘的空間也會置零的。

答:1.如果不清空,可能會在測試當中出現野值。 你做下面的試驗看看結果()

char buf[5];

cstring str,str1; //memset(buf,0,sizeof(buf)); for(int i = 0;i<5;i++) trace(「%s\r\n「,str1)

2.其實不然!特別是對於字元指標型別的,剩餘的部分通常是不會為0的,不妨作乙個試驗,定義乙個字元陣列,並輸入一串字元,如果不用memset實現清零,使用messagebox顯示出來就會有亂碼(0表示null,如果有,就預設字元結束,不會輸出後面的亂碼)

問:

如下demo是可以的,能把陣列中的元素值都設定成字元1

#include

#include

using namespace std;

int main()

而,如下程式想吧陣列中的元素值設定成1,卻是不可行的

#include

#include

using namespace std;

int main()

問題是:

1,第乙個程式為什麼可以,而第二個不行,

2,不想要用for,或是while迴圈來初始化int a[5];能做到嗎?(有沒有乙個像memset()這樣的函式初始化)

答:

1.因為第乙個程式的陣列a是字元型的,字元型佔據記憶體大小是1byte,而memset函式也是以位元組為單位進行賦值的,所以你輸出沒有問題。而第二個程式a是整型的,使用memset還是按位元組賦值,這樣賦值完以後,每個陣列元素的值實際上是0x01010101即十進位制的16843009。你看看你輸出結果是否這樣?

2.如果用memset(a,1,20);

就是對a指向的記憶體的20個位元組進行賦值,每個都用ascii為1的字元去填充,轉為二進位制後,1就是00000001,佔乙個位元組。乙個int元素是4位元組,合一起就是1000000010000000100000001,就等於16843009,就完成了對乙個int元素的賦值了。

oid *memset(void *s, int v, size_t n);

英文釋義如下:

copies the value v (converted to type unsigned char) to the first n bytes pointed to by s; returns s.

這裡s可以是陣列名,也可以是指向某一內在空間的指標;v為要填充的值;n為要填充的位元組數,通常為sizeof(s);

使用memset時要注意的是,memset是逐字節進行填充,所以s一般為char *型。對於其它型別的s,可以填充的值有兩個,0和-1。這是因為計算機中用二進位制補碼表示數字,0和二進位制補碼為全0,-1的二進位制補碼為全1。如果要初始化為其它的值,可以用乙個for迴圈,如:

for(int i=0;i

memset某個結構(或其它的資料型別)在c語言中是很常見的**,其目的是對結構(或其它的資料型別)進行初始化,通常都是將變數置為null或者0。在c++中,針對類物件除了用建構函式初始化物件外,也可以使用memset來進行初始化操作(確實有這種情況,不得已而為之,請大家不要把雞蛋砸過來!)。但是一定要注意以下所說的這種情況:如果類包含虛函式,則不能用 memset 來初始化類物件。下面是乙個例子:

class graphicsobject

virtual int area() {}

char* name()

};class circle: public graphicsobject

int area()

};void main()

結果我就不說了。因為每個包含虛函式的類物件都有乙個指標指向虛函式表(vtbl)。這個指標被用於解決執行時以及動態型別強制轉換時虛函式的呼叫問題。該指標是被隱藏的,對程式設計師來說,這個指標也是不可訪問的。當進行memset操作時,這個指標的值也要被overwrite,這樣一來,只要一呼叫虛函式,程式便崩潰。這在很多由c轉向c++的程式設計師來說,很容易犯這個錯誤,而且這個錯誤很難查。

為了避免這種情況,記住對於有虛函式的類物件,決不能使用 memset 來進行初始化操作。而是要用預設的建構函式或其它的 init 例程來初始化成員變數。

附另乙個易錯點:

memset在c中是用的非常頻繁的初始化函式了,當然也被帶到了cpp當中,因為當有如下類涉及到非常多的成員變數,很多coder經常偷懶改用memset在建構函式當中初始化

struct test

};以上如果所有成員變數是簡單的內建型別是沒有問題,但是可能某次需求迫使你需要往test中增加乙個陣列,如下

struct test

};結果在不同的編譯器是不同的,最好的情況當然是啟動時程式就掛掉了,至於為什麼會掛掉,明白vector中實現了什麼就知道了,當然不僅僅是vector,其他stl或者自己定製的容器可能都存在這個問題。

memset使用注意事項

linux下的原型宣告 name memset fill memory with a constant byte synopsis include void memset void s,int c,size t n description the memset function fills the ...

memset使用注意事項

linux下的原型宣告 name memset fill memory with a constant byte synopsis include void memset void s,int c,size t n description the memset function fills the ...

C 使用memset注意事項

include void memset void dest,int ch,std size t count 它會 轉換值 ch 為 unsigned char 並複製它到 dest 所指向物件的首 count 個位元組。因此memset方法會破壞物件內部保持狀態的私有變數,從而造成未知後果。下例對s...