欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

memset注意的地方

程序员文章站 2022-04-24 14:33:49
...
  • 先看下例子
#include <stdio.h>
#include <string.h>

void f(int *a)
{
    memset(a,0,sizeof(a));
}
int main(void)
{
    int arr[5]={1,2,3,4,5};
    f(arr); 
    for(int i=0;i<5;i++)
    {
        printf("%d ",arr[i]);
    }
    return 0;       
} 
  • 输出:

0 2 3 4 5

上面的输出竟然是 0 2 3 4 5,也就是只清空了第一个元素,这显然不是想要的结果。

为什么呢??

  • 引用别人的话来解释一下

静态数组作为参数传入某个函数的时候,就会退化成指针,也就是该数组的首地址,其数组的长度信息就丢掉了,这就是在这个语境下退化的概念。
这也是为什么在将数组作为参数传递时,同时要将数组的长度也一并传入的原因。

  • 所以将数组当作参数传递的时候,不要用memset,memset应该和数组在同一块代码区。
  • 上面例子这样改之后就可以正确输出:
#include <stdio.h>
#include <string.h>

void f(int *a,int len)
{
    memset(a,0,sizeof(int)*len);
}

int main(void)
{
    int i;
    int arr[5]={1,2,3,4,5};
    f(arr,5);   
    for(i=0;i<5;i++)
    {
        printf("%d ",arr[i]);
    }
    return 0;       
} 
  • 输出:

0 0 0 0 0

这样就正确了。

  • 用memset将 int 类型的数组置为 0 或者 -1 是可以的,但置为1或者其他数是不行的;将char数组置为任何数都是可以的。

  • 原因:

  • memset是按字节赋值,char是1字节,int是4个字节,所以char是可以的,int是不行的。(可以输出sizeof(char), sizeof(int)查看)
  • memset第一个参数是开始填充的地址,第二个参数是填充的byte,第三个参数要填充的字节数。
    比如置为1,那么赋值的结果应该是二进制数00000001000000010000000100000001,为16843009
  • 通俗的讲,就是将1变为00000001,然后从数组头开始,每8位填充一个00000001,直到填充的次数等于第三个参数。最后如果是int,那就按int的位数取出来,得到16843009。