引言
memset是计算机中C/C++语言初始化函数。作用是将某一块内存中的内容全部设置为指定的值, 这个函数通常为新申请的内存做初始化工作。
用 法: void *memset(void *s, char ch, unsigned n);
函数解释:将s中当前位置后面的n个字节 (typedef unsigned int size_t )用 ch 替换并返回 s 。
memset:作用是在一段内存块中填充某个给定的值,它是对较大的结构体或数组进行清零操作的一种最快方法。
程序示例:
#include <string.h> #include <stdio.h> #include <memory.h> int main(void) { char buffer[] = "Hello world/n"; printf("Buffer before memset: %s/n", buffer); memset(buffer, '*', strlen(buffer) ); printf("Buffer after memset: %s/n", buffer); return 0; }输出结果:
Buffer before memset: Hello world Buffer after memset: ***********
1. memset(void *s, int ch,size_t n);中ch实际范围应该在0~~255,因为该函数只能取ch的后八位赋值给你所输入的范围的每个字节,比如int a[5]赋值memset(a,-1,sizeof(int )*5)与memset(a,511,sizeof(int )*5) 所赋值的结果是一样的都为-1;因为-1的二进制码为(11111111 11111111 11111111 11111111)而511的二进制码为(00000000 00000000 00000001 11111111)后八位都为(11111111),所以数组中每个字节,如a[0]含四个字节都被赋值为(11111111),其结果为a[0](11111111 11111111 11111111 11111111),即a[0]=-1,因此无论ch多大只有后八位二进制有效,而后八位二进制的范围在(0~255)中改。而对字符数组操作时则取后八位赋值给字符数组,其八位值作为ASCII码。
2. 切记不要搞反 ch 和 n 的位置,一定要记住如果要把一个char a[20]清零,一定是 memset(a,0,20*sizeof(char));
3. 过度使用memset:这里的memset是多余的,因为这块内存马上就被全部覆盖,清零没有意义.
char buffer[4]; memset(buffer,0,sizeof(char)*4); strcpy(buffer,"123"); //"123"中最后隐藏的'\0'占一位,总长4位。
另:以下情况并不多余,因某些编译器分配空间时,内存中默认值并不为0:
char buffer[20]; memset(buffer,0,sizeof(char)*20); memcpy(buffer,"123",3); //这一条的memset并不多余,memcpy并没把buffer全部覆盖,如果没有memset, //用printf打印buffer会有乱码甚至会出现段错误。 //如果此处是strcpy(buffer,"123");便不用memset, //strcpy虽然不会覆盖buffer但是会拷贝字符串结束符
4. 指针降级错误,原因是VC函数传参过程中的指针降级,导致sizeof(a),返回的是一个something*指针类型大小的的字节数,如果是32位,就是4字节。
int some_func(struct something *a) { … … memset(a,0,sizeof(a)); … }
5. 使用memset方便的清空一个结构类型的变量或数组。
struct sample_struct { char csName[16]; int iSeq; int iType; };
对于变量 struct sample_strcut stTest;
一般情况下,清空stTest的方法:
stTest.csName[0]='/0';
stTest.iSeq=0;
stTest.iType=0;
用memset就非常方便:
memset(&stTest,0,sizeof(struct sample_struct));
如果是数组:
struct sample_struct TEST[10];
则
memset(TEST,0,sizeof(struct sample_struct)*10);