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。
上一篇: Android:页面跳转后设置不可返回