数组初始化问题:int a[3]={0}和int a[3]={1}的区别
数组定义同时初始化是个常见的问题,比如,int a[3]={0},其中a[0],a[1],a[2]这三个值都为0。
如果这样定义,同时赋值:int a[3]={1},是不是意味着a[0]=1,a[1]=1,a[2]=1呢?这是一个很容易出错的问题,答案肯定不是这样,笔者在多个编译器上试过,结果只有a[0]=1,而a[1]=0,a[2]=0。
如果要追究原因,只能去问搞编译器的人。语言有时候很难说为什么这样,或许只能方便而已。这是vc6.0下的汇编:
……
4: int a[3]={0};
00401028 mov dword ptr [ebp-0Ch],0 ;a[0]=0
0040102F xor eax,eax ;寄存器eax清零,eax的值为0
00401031 mov dword ptr [ebp-8],eax ;把eax的值赋给a[1],所以a[1]=0
00401034 mov dword ptr [ebp-4],eax;把eax的值赋给a[2],所以a[1]=0
5: int b[3]={1};
00401037 mov dword ptr [ebp-18h],1;b[0]=1
0040103E xor ecx,ecx;寄存器ecx清零, ecx的值为0
00401040 mov dword ptr [ebp-14h],ecx;把ecx的值赋给b[1],所以b[1]=0
00401043 mov dword ptr [ebp-10h],ecx;把ecx的值赋给b[2],所以b[1]=0
int数组为什么一般不能用memset初始化每个元素
对于整数型数组,若要是初始化为非0的值时应该使用for循环而对于字符型 或 0 应该使用memset 函数
如下是可以的,能把数组中的元素值都设置成字符1,
- #include <iostream>
- #include <cstring>
- using namespace std;
- int main()
- {
- char a[5];
- memset(a,'1',5);
- for(int i = 0;i < 5;i++)
- cout<<a[i]<<" ";
- system("pause");
- return 0;
- }
而,如下程序想吧数组中的元素值设置成1,却是不可行的
- #include<iostream>
- #include <cstring>
- using namespace std;
- int main()
- {
- int a[5];
- memset(a,1,5);//这里改成memset(a,1,5 *sizeof(int))也是不可以的
- for(int i = 0;i < 5;i++)
- cout<<a[i]<<" ";
- system("pause");
- return 0;
- }
问题是:
1,第一个程序为什么可以,而第二个不行,
2,不想要用for,或是while循环来初始化inta[5];能做到吗?(有没有一个像memset()这样的函数初始化)
答:
1.因为第一个程序的数组a是字符型的,字符型占据内存大小是1Byte,而memset函数也是以字节为单位进行赋值的,所以你输出没有问题。而第二个程序a是整型的,使用memset还是按字节赋值,这样赋值完以后,每个数组元素的值实际上是0x01010101即十进制的16843009。你看看你输出结果是否这样?
2.如果用memset(a,1,20);
就是对a指向的内存的20个字节进行赋值,每个都用ASCII为1的字符去填充,转为二进制后,1就是00000001,占一个字节。一个INT元素是4字节,合一起就是1000000010000000100000001,就等于16843009,就完成了对一个INT元素的赋值了。
如何给数组A初始化为无穷大?
利用上面原理,memset( A, 0x3f,sizeof( A) ),0x3f 转换成十进制为63,每个字节都用ASCII码为63的字符去填充,转换成为禁止就是00111111,占一个字节,int型4个字节就是 00111111001111110011111100111111,约等于无穷大