关于数组名,数组名是首元素地址,但有两个例外,sizeof单独放数组名,表示计算整个数组的大小,单位是字节。
&数组名,表示取出整个数组的地址,除了这两个例外,你见到的其他数组名都是首元素的地址。
int arr[10]={0};
大家看,数组名和&数组名打印出来的值是一样的,那为什么还要区分开来呢,原因是arr它的类型是int*的,而&arr它的类型是int(*)[10],如果只是打印首地址 那么是一样的,但是让它们+1的话,访问的步长就不一样了,arr+1访问的步长是int,是四个字节,但&arr+1访问的步长是整个数组,也就是10*int,40个字节,代码如下:
下面让我们来看一下sizeof的使用:
#include<stdio.h>
int main()
{
int a[] = { 1,2,3,4 };
printf("%d\n", sizeof(a));
//16,a作为数组名单独放在sizeof内部,计算的是数组的总大小,单位是字节
printf("%d\n", sizeof(a + 0));
//a并非单独放在sizeof内部,也没有&,所以数组名a就是数组首元素的地址
//a+0还是数组首元素的地址,是地址大小就是 4/8 个字节
printf("%d\n", sizeof(*a));
//a是首元素的地址,*a就是首元素,sizeof(*a)就算的就是首元素的大小 - 4
//a - int*
//*a - int
printf("%d\n", sizeof(a + 1));
//a是首元素的地址,a+1是第二个元素的地址,sizeof(a+1)计算的是指针的大小 - 4/8
//a - int*
//a+1, 跳过一个int
printf("%d\n", sizeof(a[1]));
//a[1]就是数组的第二个元素,sizeof(a[1])的大小 - 4个字节
printf("%d\n", sizeof(&a));
//&a取出的数组的地址,数组的地址,也是地址呀,sizeof(&a)就是 4/8 个字节
printf("%d\n", sizeof(*&a));
//&a是数组的地址,是数组指针类型,*&a是都数组指针解引用,访问一个数组的大小
//16字节
//sizeof(*&a) ==> sizeof(a) =16
printf("%d\n", sizeof(&a + 1));
//&a数组的地址,&a+1跳过整个数组,&a+1还是地址,是 4/8 个字节
printf("%d\n", sizeof(&a[0]));
//a[0]是数组的第一个元素,&a[0]是第一个元素的地址,是 4/8 个字节
printf("%d\n", sizeof(&a[0] + 1));
//&a[0]是第一个元素的地址,&a[0]+1就是第二个元素的地址,是 4/8 个字节
return 0;
}
总结一下,sizeof计算的是对象所占的空间大小,单位是字节,
sizeof是单目操作符,不是库函数。
接下来看strlen:strlen是针对字符串的,它的功能是计算字符串的长度,strlen需要一个地址,从首地址一直往后数,直至遇到\0.
#include<string.h>
#include<stdio.h>
int main()
{
char arr[] = { 'a','b','c','d','e','f' };
printf("%d\n", strlen(arr));
//随机值,arr是数组名,但是没有放在sizeof内部,也没&,arr就是首元素的地址
//strlen得到arr后,从arr数组首元素的地方开始计算字符串的长度,直到直到\0,但是arr数组中没有\0,arr内存的后边是否有\0,在什么位置
//是不确定的,所以\0之前出现了多少个字符是随机的。
printf("%d\n", strlen(arr + 0));
//arr是数组首元素的地址,arr+0还是首元素的地址
//随机值
printf("%d\n", strlen(*arr));
//arr是数组首元素的地址,*arr 是首元素 - ‘a’ - 97
//strlen就把‘a’的ASCII码值 97 当成了地址
//err 会非法访问内存
printf("%d\n", strlen(arr[1]));
//arr[1] - 'b' - 98 - err
printf("%d\n", strlen(&arr));
//随机值,&arr是数组的地址,数组的地址也是指向数组起始位置,和第一个案例一样
printf("%d\n", strlen(&arr + 1));//随机值
printf("%d\n", strlen(&arr[0] + 1));//随机值
return 0;
}
#include<stdio.h>
#include<string.h>
int main()
{
char arr[] = "abcdef";
printf("%d\n", strlen(arr));//6
//arr是首元素地址,地址开始往后数,直至遇到\0
printf("%d\n", strlen(arr + 0));//6
//arr是首元素地址,首元素地址+0,还是原来的位置,所以也是6
printf("%d\n", strlen(*arr));
//err
//strlen需要的是一个地址,*arr==‘a’,所以编译器会报错
printf("%d\n", strlen(arr[1]));
//err
strlen需要的是一个地址,arr[1]==‘b’,所以编译器会报错
printf("%d\n", strlen(&arr));
//6
//&arr是首元素地址,地址开始往后数,直至遇到\0
printf("%d\n", strlen(&arr + 1));
//随机值
//&arr+1,跳过了整个字符串,也就是从‘f’后边的元素开始数,直至遇到\0,
//但后面是什么我们压根不知道,所以是随机值
printf("%d\n", strlen(&arr[0] + 1));
//5
//&arr[0]+1==取出‘a’的地址然后向后走一步到‘b’,从b的地址开始数,直至遇到\0,所以是5
return 0;
}