memset注意的地方

  • 先看下例子
#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。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • __block和__weak修饰符的区别其实是挺明显的:1.__block不管是ARC还是MRC模式下都可以使用,...
    LZM轮回阅读 8,687评论 0 6
  • 1.写一个NSString类的实现 +(id)initWithCString:(c*****t char *)nu...
    韩七夏阅读 9,166评论 2 37
  • iOS面试小贴士 ———————————————回答好下面的足够了------------------------...
    不言不爱阅读 6,182评论 0 7
  • 多线程、特别是NSOperation 和 GCD 的内部原理。运行时机制的原理和运用场景。SDWebImage的原...
    LZM轮回阅读 6,086评论 0 12
  • ———————————————回答好下面的足够了---------------------------------...
    恒爱DE问候阅读 5,682评论 0 4