1 数组的概念
- 数组是相同类型的变量的有序集合
- 数组中的元素没有名字
如下图是一个数组:
1.1 数组的大小
- 数组的大小是数组元素个数乘以元素的数据类型
- 数组在一片连续的内存空间中存储元素
- 数组元素的个数可以显示或者隐式的指定
那么数组a[0]=1,a[1]=2,a[2]=0,a[3]=0,a[4]=0
数组b的元素个数是2,b[0]=1,b[1]=2.
sizeof(a) = 20,sizeof(b)=8
数组的元素的个数可以使用下面的公式求解:
cnt = sizeof(array)/sizeof(type)
想要将数组全部初始化为0,直接写:
int a[5] = {0};
1.2 数组的初始化
如下代码27-1.c:
#include <stdio.h>
int main()
{
int a[5] = {1, 2};
int b[] = {1, 2};
printf("a[2] = %d\n", a[2]);
printf("a[3] = %d\n", a[3]);
printf("a[4] = %d\n", a[4]);
printf("sizeof(a) = %d\n", sizeof(a));
printf("sizeof(b) = %d\n", sizeof(b));
printf("count for a: %d\n", sizeof(a)/sizeof(int));
printf("count for b: %d\n", sizeof(b)/sizeof(int));
return 0;
}
- 编译运行结果为:
2 数组的地址与数组名
- 数组名代表数组首元素的地址,但是如果想要获取数组的地址,还是需要加上&地址获取符。如&a是数组a的地址。这里有一点不能绕晕了,如果a是数组,那么a的值与&a的值是相等的。这与普通变量是不一样的。
- 整个数组的地址值等于数组第一个元素的地址,它们在数值上相等,但是意义不一样。
如下代码27-2.c:
#include <stdio.h>
int main()
{
int a[5] = { 0 };
printf("a = %p\n", a);
printf("&a = %p\n", &a);
printf("&a[0] = %p\n", &a[0]);
return 0;
}
运行结果如下:
注意:
如果想获取整个数组的地址,只能这样获取:int* p =&a 。那么p就是数组的地址,p+1就是数组最后一个元素的下一个地址。
但是如果直接用int *p = a, p+1就是数组的第二个元素的地址。对于&a[0]也是一样的
为什么会不同的呢?实际上可以这么理解:
- 数组名a只是代表数组首元素的地址值,它不包含数组的长度信息,所以a+1就直接相当于数组的第二个元素的地址
- 而&a代表整个数组的地址,它包含了数组的长度信息,所以在&a + 1时,相当于直接越过了整个数组的长度,调到数组的末尾的下一个元素的地址。
3 数组名不能作为左值使用
- 数组与指针是不同的。
这个话题,会在后面有专门的两篇文章学习。所以这里只是简单说一点:
- 在表达式中,数组名只能作为右值使用,不能作为左值使用
#include <stdio.h>
int main()
{
int a[5] = {0};
int b[2];
int* p = NULL;
p = a;
printf("a = %p\n", a);
printf("p = %p\n", p);
printf("&p = %p\n", &p);
printf("sizeof(a) = %d\n", sizeof(a));
printf("sizeof(p) = %d\n", sizeof(p));
printf("\n");
p = b;
printf("b = %p\n", b);
printf("p = %p\n", p);
printf("&p = %p\n", &p);
printf("sizeof(b) = %d\n", sizeof(b));
printf("sizeof(p) = %d\n", sizeof(p));
b = a;
return 0;
}
上述代码编译会出错:
将第27行注释掉,就可以编译运行:
这说明,在表达式中,数组名不能作为左值使用
4 总结
- 数组是一片连续的内存空间
- 数组的地址 和 数组首元素的地址 意义不同
- 数组名不能作为左值使用