int a[10];
memset(a,0,sizeof(a));//ok
memset(a,-1,sizeof(a));//ok
menset(a,-1,sizeof(char)*40);//等价于
memset(a,-1,sizeof(int)*10);//等价于
分析原因:
在计算机内数据以补码保存;
负数的补码算法:
- 求出负数正数的补码,全部取反,再+1
比如 -1:-1的正数是1;
0000 0001 -> 1111 1110 -> 1111 1111 - 求出负数正数的补码,先减1,再全部取反
比如-1:-1的正数是1
0000 0001-> 0000 0000 -> 1111 1111
memset是按照Byte来填充二进制的
1 Byte = 8 bit
1 char = 1 Byte
1 int = 4 Byte = 4 char = 32 bit
如图:
#include <iostream>
#include <cstring>
#include <stdio.h>
#include <stdlib.h>
using namespace std;
void print(int arr[],int len){
for(int i=0;i<len;++i){
printf("%d ",arr[i]);
}
printf("\n");
}
int main(){
int arr[10];
memset(arr,0,sizeof(int)*10);
memset(arr,-1,sizeof(char)*10);
//memset(arr,-1,sizeof(char)*40);//全部是-1
print(arr,10);
return 0;
}
输出结果:
-1 -1 65535 0 0 0 0 0 0 0
对memset(arr,-1,sizeof(char)*10);
1111 1111
有10个
arr[0]:
1111 1111
1111 1111
1111 1111
1111 1111
由于是负数,需要转换成原码;
符号位不变,-1,再按位取反:1000 0000
0000 0000
0000 0000
0000 0001
换算成int,就是-1,所以arr[0] = -1
arr[1]:
类推出arr[1] = -1
arr[2]:
此时还剩下2个 1111 1111
arr[2]的二进制如下:
0000 0000
0000 0000
1111 1111
1111 1111
(不要写成 1111 1111
1111 1111
0000 0000
0000 0000
)
换成16进制:0x00 00 FF FF
换算成int,就是65535
从arr[3]以后,memset都没填充,所以都还是0
。
为什么memset只能给int数组赋值0或-1,而不支持其他的数字?
因为0补码是32个0,-1补码是32个1。看到这里应该可以理解了。