美文网首页
memset函数

memset函数

作者: 霜之幽语 | 来源:发表于2018-02-23 16:17 被阅读53次

    将s所指向的某一块内存中的前n个字节的内容全部设置为ch指定的ASCII值, 第一个值为指定的内存地址,块的大小由第三个参数指定,这个函数通常为新申请的内存做初始化工作, 其返回值为指向s的指针

    void *memset(void *s, int ch, size_t n);
    

    函数解释:将s中当前位置后面的n个字节 (typedef unsigned int size_t )用 ch 替换并返回 s 。
    memset:作用是在一段内存块中填充某个给定的值,它是对较大的结构体数组进行清零操作的一种最快方法。

    常见错误

    1. memset函数按字节对内存块进行初始化,所以不能用它将int数组初始化为0和-1之外的其他值(除非该值高字节和低字节相同)。
    2. memset(void s, int ch,size_tn);中key实际范围应该在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多大只有后八位二进制有效,而八位二进制[2] 的范围(0~255)YKQ改。而对字符数组操作时则取后八位赋值给字符数组,其八位值作为ASCII[3] 码。
    3. 搞反了 ch 和 n 的位置.
      一定要记住如果要把一个char a[20]清零,一定是 memset(a,0,20*sizeof(char));

    而不是 memset(a,20*sizeof(char),0);

    1. 过度使用memset,我想这些程序员可能有某种心理阴影,他们惧怕未经初始化的内存,所以他们会写出这样的代码:
    char buffer[4];
    memset(buffer, 0, sizeof`(char)*4);
    strcpy(buffer, "123");
    //"123"中最后隐藏的'\0'占一位,总长4位。
    

    这里的memset是多余的. 因为这块内存马上就被全部覆盖,清零没有意义.

    另:以下情况并不多余,因某些编译器分配空间时,内存中默认值并不为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但是会拷贝字符串结束符
    
    1. 其实这个错误严格来讲不能算用错memset,但是它经常在使用memset的场合出现
    int some_func(struct something *a)
    {
    …
    …
    memset(a,0,sizeof(a));
    …
    }
    

    这里错误的原因是VC函数传参过程中的指针降级,导致sizeof(a),返回的是一个something指针类型大小的字节数*,如果是32位,就是4字节。
    参考百度百科词条

    相关文章

      网友评论

          本文标题:memset函数

          本文链接:https://www.haomeiwen.com/subject/asuxxftx.html