竞赛中经常使用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位就是0000001
最终int类型数组一个元素的内容就变成了 0000001000000100000010000001
明显是个很大的数。
那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
看一下INF的二进制表示:
00111111001111110011111100111111
可以看出低8位是00111111,也就是实际赋值时是用
00111111来填充一个int类型,结果正好也是
00111111001111110011111100111111
于是就成功用memset进行了赋值。