memset函数的使用原理

memset函数的使用原理

2017年02月01日 20:35:44 catTom 阅读数:2367更多

注:带颜色的文本是转发者加的

个人分类: Linuxacm

竞赛中经常使用memset函数进行数组初始化,但只能初始为0, -1。哪怕写成:

 
  1. int num[N];

  2. memset(num, 1, sizeof(num));

num[0]也是个很大的数。

今天看了一下C标准库里面memset函数的源码,一切就清楚了

 
  1. void *(memset)(void *s, int c, size_t n)

  2. {

  3. const unsigned char uc = c;

  4. unsigned char *su;

  5. for (su = s; 0 < n; ++su, --n)

  6. *su = uc;

  7. return (s);

  8. }

可以看出这个函数本身就是为字符数组设计的。

源码说明对于数组参数是按照字符数组的形式进行赋值的。

也就是c的值每次赋值给一个字节,而不是4个字节,c的类型是int类型,4个字节,

但只会赋值最低位的一个字节的内容。

当c是1的时候,二进制低8位就是00000001

最终int类型数组一个元素的内容就变成了 00000001000000010000000100000001

明显是个很大的数。


那0跟-1为什么可以呢?

因为0的二进制表示全都是0,-1的二进制表示全都是1,

全0填充32位,int类型也是0

全1填充32位,int类型也是-1

知道了这个内部原理就可以利用一下了,

 
  1. #define INF 0x3f3f3f3f

  2. int num[N];

  3. memset(num, INF, sizeof(num));

这样得到的数组里面的每一个元素数值都是INF

说明:要给一个int类型的数组赋值,要保证赋值的数譬如是n赋值给num,结果num的值和n的值是一样的话,必须n的值是一个四字节的数,并且每个字节的值必须相同才行,譬如赋值为0x3f3f3f3f,或0x78787878,因为赋值时是到n中的取四个字节中最后一个字节78后按照sizeof(num)的值的次数去填充到num中,如果sizeof()的值是4次,则填充4次,如果sizeof()的值是7,则填充7次,但是去访问int类型的num时,是按照4字节去取值的,也就说取了4个78组成的四字节来得到sum的值。如果n的值不是0x3f3f3f3f,这样排列,譬如是1,则会打错而特错。(因为memset()函数是 按照一个字节一个字节的赋值的。)

看一下INF的二进制表示:

00111111001111110011111100111111

可以看出低8位是00111111,也就是实际赋值时是用

00111111来填充一个int类型,结果正好也是

00111111001111110011111100111111

于是就成功用memset进行了赋值。

猜你喜欢

转载自blog.csdn.net/weixin_41632560/article/details/86140226