競賽中經常使用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進行了賦值。