一、数组名的理解:数组名是数组首元素的地址。但是有两个例外:(1)sizeof(数组名)例如sizeof(arr),此时的数组名表示整个数组,计算的整个数组的大小,单位是字节;(2)&数组名(例如&arr),这里的数组名表示整个数组,&数组名取出的数数组的地址。
二、举例:
1、一维数组:
(1)、
int main()
{
int a[] = { 1,2,3,4 };
printf("%d\n", sizeof(a));//16,数组名a单独放在sizeof内部,数组名表示整个数组,计算的是整个数组的大小单位是字节,是16字节
printf("%d\n", sizeof(a + 0));//4或8,a并非单独放在sizeof内部,也没有&,所以数组名a是数组首元素的地址,a+0还是首元素的地址
printf("%d\n", sizeof(*a));//4,a并非单独放在sizeof内部,也没有&,所以数组名a是数组首元素的地址//*a 就是 首元素,大小就是4Byte *a == *(a+0) == a[0]
printf("%d\n", sizeof(a + 1));//4或8,a并非单独放在sizeof内部,也没有&,所以数组名a是数组首元素的地址,a+1就是第二个元素的地址 a+1 == &a[1] 是第2个元素的地址,是地址就是4/8个字节
printf("%d\n", sizeof(a[1]));//4,a[1]就是数组的第二个元素,这里计算的就是第二个元素的大小,单位是字节
printf("%d\n", sizeof(&a));//4或8,&a - 是取出数组的地址,但是数组的地址也是地址,是地址就是4/8个Byte 数组的地址 和 数组首元素的地址 的本质区别是类型的区别,并非大小的区别
printf("%d\n", sizeof(*&a));//16, 对数组指针解引用访问一个数组的大小,单位是字节
//sizeof(*&a) --- sizeof(a) //16
printf("%d\n", sizeof(&a + 1));//4或8,&a数组的地址,&a+1还是地址,是地址就是4/8个字节
printf("%d\n", sizeof(&a[0]));//4或8,&a[0]是首元素的地址, 计算的是地址的大小 4/8 个字节
printf("%d\n", sizeof(&a[0] + 1));//4或8,&a[0]是首元素的地址,&a[0]+1就是第二个元素的地址,大小4/8个字节
return 0;
}
指针变量的大小和类型无关,不管什么类型的指针变量,大小都是4/8个字节
指针变量是用来存放地址的,地址存放需要多大空间,指针变量的大小就是几个字节
32位环境下,地址是32个二进制位,需要4个字节,所以指针变量的大小就是4个字节
64位环境下,地址是64个二进制位,需要8个字节,所以指针变量的大小就是8个字节
(2)、字符数组:strlen求字符串长度——统计字符串\0之前字符的个数
#include <string.h>
int main()
{
//字符数组
char arr[] = { 'a','b','c','d','e','f' };//6
printf("%d\n", strlen(arr));//随机值,arr是首元素的地址
printf("%d\n", strlen(arr + 0));//随机值,arr是首元素的地址, arr+0还是首元素的地址
printf("%d\n", strlen(*arr));//err,arr是首元素的地址, *arr就是首元素 - 'a' - 97,站在strlen的角度,认为传参进去的'a'-97就是地址,97作为地址,直接进行访问,就是非法访问
printf("%d\n", strlen(arr[1]));//err, 'b' - 98
printf("%d\n", strlen(&arr));//随机值
printf("%d\n", strlen(&arr + 1));//随机值
printf("%d\n", strlen(&arr[0] + 1));//随机值
return 0;
}
(3)、
int main()
{
char arr[]="abcdef";
printf("%d\n", sizeof(arr));//7,sizeof(数组名)计算的是整个数组的大小,字符串加上\0共七个元素,每个元素一个字节(char类型)
printf("%d\n", sizeof(arr + 0));//4或者8,arr并没有单独放在sizeof内部,所以表示首元素地址,(arr+0)还是表示首元素地址;
printf("%d\n", sizeof(*arr));//1,arr没有单独放在sizeof内部,前面也没有&符号,所以arr表示首元素地址,*arr是对arr解引用,就是首元素a,就是一个字节;
printf("%d\n", sizeof(arr[1]));//1,arr[1]表示第二个元素,所以是一个字节
printf("%d\n", sizeof(&arr));//4或者8,&arr表示整个数组的地址,是地址就是4/8字节
printf("%d\n", sizeof(&arr + 1));//4或者8,&arr+1指向数组最末尾之后的位置,也是地址,是地址就是4或者8字节
printf("%d\n", sizeof(&arr[0] + 1)); //4或者8,&arr[0]+1表示第二个元素地址,所以是4或者8字节
printf("%d\n", strlen(arr));//6,数组名没有单独放在sizeof内部,前面也没有&符号,所以arr表示首元素地址,所以从首元素开始数
printf("%d\n", strlen(arr + 0));//6,数组名没有单独放在sizeof内部,前面也没有&符号,所以arr表示首元素地址,arr+0表示首元素地址,所以从首元素开始数
printf("%d\n", strlen(*arr));//err,*arr对首元素地址解引用,就是a,所以传进97,会报错;
printf("%d\n", strlen(arr[1]));//err,arr[1]表示第二个元素,也就是b,所以传入98,会报错
printf("%d\n", strlen(&arr));//6,&arr取出的数组的地址,会从数组首元素开始往后数
printf("%d\n", strlen(&arr + 1));//随机值,&arr+1表示跳过arr数组后一个位置,不知道什么时候会遇到\0
printf("%d\n", strlen(&arr[0] + 1));//5,&arr[0]表示首元素,&arr[0]+1表示第二个元素,从第二个元素开始数
return 0;
}
(4)、
int main()
{
char* p = "abcdef";
printf("%d\n", strlen(p));//6,p指向a的地址,即从a开始往后数
printf("%d\n", strlen(p + 1));//5,p+1是指向第二个元素的地址,即b的地址
printf("%d\n", strlen(*p));//err,*p是对a的地址解引用,就是元素a,即97
printf("%d\n", strlen(p[0]));//err,p[0]还是表示首元素,即a
printf("%d\n", strlen(&p));//随机值,&p是不知道从哪里开始数,p是指向a的地址,&p是在a之前某个位置
printf("%d\n", strlen(&p + 1));//随机值,&p+1是把p的地址跳过去
printf("%d\n", strlen(&p[0] + 1));//5,&p[0]是首元素地址,再+1是第二个元素地址,所以从第二个元素开始数
printf("%d\n", sizeof(p));//4或者8 计算的是指针变量的大小
printf("%d\n", sizeof(p + 1));//4或者8,p+1还是地址,大小是4/8个字节
printf("%d\n", sizeof(*p));//1个字节, *p == 'a'
printf("%d\n", sizeof(p[0]));//1个字节, p[0]--> *(p+0) --> *p == 'a';
printf("%d\n", sizeof(&p));//4/8个字节,&p 是地址
printf("%d\n", sizeof(&p + 1));//4或者8,&p是地址,&p+1还是地址,是地址就是4/8个字节
printf("%d\n", sizeof(&p[0] + 1));//4或者8,表示第二个元素的地址
return 0;
}
2、二维数组
int main()
{
int a[3][4] = { 0 };
printf("%zd\n", sizeof(a));//48,-数组名a单独放在了sizeof内存,表示整个数组,sizeof(a)计算的是数组的大小,单位是字节
printf("%zd\n", sizeof(a[0][0]));//4,-a[0][0]是数组的第一行第一个元素,这里计算的就是一个元素的大小,单位是字节
printf("%zd\n", sizeof(a[0]));//16, - a[0]是第一行这个一维数组的数组名,数组名单独放在了sizeof内部,a[0]就表示整个第一行这个一维数组,sizeof(a[0])计算的整个第一行这个一维数组的大小
printf("%zd\n", sizeof(a[0] + 1));//4或者8, - a[0]并非单独放在sizeof内部,也没有&,所以a[0]表示第一行这个一维数组首元素的地址也就是第一行第一个元素的地址
printf("%zd\n", sizeof(*(a[0] + 1)));//4, - a[0] + 1是第一行第二个元素的地址,*(a[0] + 1))就是第一行第二个元素
printf("%zd\n", sizeof(a + 1));//4或者8,a 作为二维数组的数组名,并没有单独放在sizeof内部,也没有&,a就是数组首元素的地址,也就是第一行的地址, a 的类型是 int(*)[4],a+1 就是第二行的地址,类型是:int(*)[4]
printf("%zd\n", sizeof(*(a + 1)));//16, a+1是第二行的地址,*(a+1)就是第二行,计算的就是第二行的大小;另外一个角度理解:*(a+1) -- a[1],sizeof(a[1]) - a[1]这个第二行的数组名,单独放在了sizeof内部,计算的是第二行的大小
printf("%zd\n", sizeof(&a[0] + 1));//4或者8,a[0]是第一行的数组名,&a[0]取出的是数组的地址,取出的是第一行这个一维数组的地址,类型就是int(*)[4],&a[0]+1 就是第二行的地址,类型就是int(*)[4]
printf("%zd\n", sizeof(*(&a[0] + 1)));//16,*(&a[0] + 1)得到的就是第二行,计算的就是第二行的大小
printf("%zd\n", sizeof(*a));//16,a表示数组首元素的地址,也就是第一行的地址,*a 就是第一行,也就相当于是第一行的数组名
printf("%zd\n", sizeof(a[3]));//16,不会越界
return 0;
}