3. 二维数组
3.1 思考该结果是什么?
int main() { int a[3][4] = { 0 }; printf("%d\n", sizeof(a)); printf("%d\n", sizeof(a[0][0])); printf("%d\n", sizeof(a[0])); printf("%d\n", sizeof(a[0] + 1)); printf("%d\n", sizeof(*(a[0] + 1))); printf("%d\n", sizeof(a + 1)); printf("%d\n", sizeof(*(a + 1))); printf("%d\n", sizeof(&a[0] + 1)); printf("%d\n", sizeof(*(&a[0] + 1))); printf("%d\n", sizeof(*a)); printf("%d\n", sizeof(a[3])); return 0; }
详细讲解:
int main() { int a[3][4] = { 1,2,3,4,5,6,7,8,9,10,11,12 }; printf("%d\n", sizeof(a));//48 - a这个二维数组的数组名单独放在sizeof内部,计算整个数组的大小 printf("%d\n", sizeof(a[0][0]));//第一行第一个元素,4个字节 printf("%d\n", sizeof(a[0]));//16 //a[0] 第一行的数组名,这时数组名单独放在sizeof内部了 //计算的是第一行数组的大小,单位是字节,16 printf("%d\n", sizeof(a[0] + 1));//4 //a[0]不是单独放在sizeof内部,a[0]表示的首元素的地址,即第一行第一个元素的地址 - &a[0][0] //a[0] + 1 是第一行第2个元素的地址 &a[0][1] printf("%d\n", sizeof(*(a[0] + 1)));//a[0][1] 大小是:4个字节 printf("%d\n", sizeof(a + 1));// //a作为二维数组的数组名并非单独放在sizeof内部,所以表示首元素的地址 //二维数组的首元素是第一行,这里的a就是第一行的地址--- int (*)[4]数组指针 //a+1是跳过第一行,指向了第二行 printf("%d\n", sizeof(*(a + 1)));//16 //*(a+1)-->a[1] printf("%d\n", sizeof(&a[0] + 1));//4/8 //&a[0]是第一行的地址 //&a[0]+1是第二行的地址 printf("%d\n", sizeof(*(&a[0] + 1)));//16 a[1] printf("%d\n", sizeof(*a));//16 *a - 就是第一行 //*a -- *(a+0) -- a[0] printf("%d\n", sizeof(*arr+1);//4 /8 //*arr--arr[0],arr[0]+1 = &arr[0][0]+1-->&arr[0][1] printf("%d\n", sizeof(a[3]));//16 并不会越界 //不会越界的原因: int a = 5; short s = 11; printf("%d\n", sizeof(s = a + 2));//发生截断,2 printf("%d\n", s);//按理说结果为7但真正结果为11,因为sizeof内部不会真正计算 return 0; }
图片讲解:
3.2 不会越界的原因
不会越界的原因:
int a = 5; short s = 11; printf("%d\n", sizeof(s = a + 2));//发生截断,2 printf("%d\n", s);//按理说结果为7但真正结果为11,因为sizeof内部不会真正计算
sizeof 在编译过程的时候就处理掉了,不会在进行后面的流程了,分为值属性和类型属性,值属性是7,类型属性为2,类型属性先被计算,所以不会产生越界访问。
4. 总结
数组名的意义:
sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
&数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
除此之外所有的数组名都表示首元素的地址。
二维数组的首元素是第一行。
如果这份博客对大家有帮助,希望各位给恒川一个免费的点赞作为鼓励,并评论收藏一下,谢谢大家!!!
制作不易,如果大家有什么疑问或给恒川的意见,欢迎评论区留言。