memset函数的使用原理
2017年02月01日 20:35:44 catTom 阅读数:2367更多
注:带颜色的文本是转发者加的
竞赛中经常使用memset函数进行数组初始化,但只能初始为0, -1。哪怕写成:
-
int num[N];
-
memset(num, 1, sizeof(num));
num[0]也是个很大的数。
今天看了一下C标准库里面memset函数的源码,一切就清楚了
-
void *(memset)(void *s, int c, size_t n)
-
{
-
const unsigned char uc = c;
-
unsigned char *su;
-
for (su = s; 0 < n; ++su, --n)
-
*su = uc;
-
return (s);
-
}
可以看出这个函数本身就是为字符数组设计的。
源码说明对于数组参数是按照字符数组的形式进行赋值的。
也就是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
知道了这个内部原理就可以利用一下了,
-
#define INF 0x3f3f3f3f
-
int num[N];
-
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进行了赋值。