memset函數的使用原理

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




發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章