指针和数组笔试题解析
看完我之前的博客后,大家对指针和数组应该有了比较清晰的认知
现在来看看几道笔试题
首先让我们复习一下数组名
对数组名的理解
数组名是数组首元素的地址
有且仅有两个例外
1.sizeof(数组名),这里的数组名不是数组首元素的地址,这里表示整个数组,sizeof(数组名)计算的是整个数组的大小,单位是字节
2.&数组名,这里的数组名表示整个数组,&数组名表示的是整个数组的地址
一维数组
//一维数组
int a[] = {1,2,3,4};
printf("%d\n",sizeof(a)); //4*4=16
printf("%d\n",sizeof(a+0)); //a+0还是数组首元素的地址,地址大小为4/8
printf("%d\n",sizeof(*a));//*a为首元素,大小为4个字节
printf("%d\n",sizeof(a+1));//a+1是第二个元素的地址,地址大小为4/8个字节
printf("%d\n",sizeof(a[1]));//第二个元素的大小就是4个字节
printf("%d\n",sizeof(&a));//&a为数组的地址,数组的地址也是地址,4/8个字节
printf("%d\n",sizeof(*&a));//即整个数组的大小,16字节
printf("%d\n",sizeof(&a+1));//跳过了整个数组,因为还是地址,4/8字节
printf("%d\n",sizeof(&a[0]));//&a[0]是首元素的地址,4/8字节
printf("%d\n",sizeof(&a[0]+1));//第二个元素的地址,4/8字节
图解如下
注释:只要是指针的大小就是4/8字节(x86平台上是4字节,x64平台上是8字节)
字符数组
//字符数组
char arr[] = {'a','b','c','d','e','f'};
printf("%d\n", sizeof(arr));//数组名单独放在sizeof内部,这里的arr表示整个数组的大小,单位是字节,总共6个字节
printf("%d\n", sizeof(arr+0));//数组首元素的地址,4/8字节
printf("%d\n", sizeof(*arr));//*arr是首元素,大小1个字节
printf("%d\n", sizeof(arr[1]));//第二个元素大小1字节
printf("%d\n", sizeof(&arr));//数组的地址,4/8字节
printf("%d\n", sizeof(&arr+1));//跳过整个数组的地址,4/8字节
printf("%d\n", sizeof(&arr[0]+1));第二个元素的地址,4/8字节
printf("%d\n", strlen(arr));//因为字符数组arr中没有\0,所以在求字符串长度的时候,会一直往后找,产生的结构就是随机值
printf("%d\n", strlen(arr + 0));//arr + 0是首元素的地址,和第一个一样,也是随机值
printf("%d\n", strlen(*arr));//err, arr是数组首元素的地址,*arr就是数组首元素,就是'a'-97
//strlen函数参数的部分需要传一个地址,当我们传递的是'a'时,'a'的ASCII码值是97,那就是将97作为地址传参
//strlen就会从97这个地址开始统计字符串长度,这就非法访问内存了
printf("%d\n", strlen(arr[1]));//err
printf("%d\n", strlen(&arr));//&arr是数组的地址,数组的地址和数组首元素的地址,值是一样的,那么传递给strlen函数后,依然是从数组的第一个元素的位置开始往后统计
printf("%d\n", strlen(&arr + 1));//随机值
printf("%d\n", strlen(&arr[0] + 1));//&arr[0] + 1是第二个元素的地址。结果也是随机值
char arr[] = "abcdef";//[a b c d e f \0]
printf("%d\n", strlen(arr));//6
printf("%d\n", strlen(arr + 0));//6
printf("%d\n", strlen(*arr));//err
printf("%d\n", strlen(arr[1]));//err
printf("%d\n", strlen(&arr));//6
printf("%d\n", strlen(&arr + 1));//随机值
printf("%d\n", strlen(&arr[0] + 1));//5
printf("%d\n", sizeof(arr));//7
//char [7]
printf("%d\n", sizeof(arr + 0));//arr + 0是首元素的地址
printf("%d\n", sizeof(*arr));//*arr其实就是首元素,1个字节
//*arr--> *(arr+0) -- arr[0]
printf("%d\n", sizeof(arr[1]));//arr[1]是第二个元素,1个字节
printf("%d\n", sizeof(&arr));//&arr是数组的地址,是地址就是4/8个字节
printf("%d\n", sizeof(&arr + 1));//&arr + 1是跳过一个数组的地址,4/8
printf("%d\n", sizeof(&arr[0] + 1));//&arr[0] + 1是第二个元素的地址 4/8
int main()
{
char* p = "abcdef";
// 012345
printf("%d\n", strlen(p));//6
printf("%d\n", strlen(p + 1));//5
//printf("%d\n", strlen(*p));//err
//printf("%d\n", strlen(p[0]));//err
printf("%d\n", strlen(&p));//随机值
printf("%d\n", strlen(&p + 1));//随机值
printf("%d\n", strlen(&p[0] + 1));//5
printf("%d\n", sizeof(p));//p是一个指针变量
大小就是4/8
printf("%d\n", sizeof(p + 1));//p+1是'b'的地址,是地址大小就是4/8个字节
printf("%d\n", sizeof(*p));//*p 就是'a',就是1个字节
printf("%d\n", sizeof(p[0]));//p[0]--> *(p+0) --> *p 1个字节
printf("%d\n", sizeof(&p));//4/8
//&p -- char**
printf("%d\n", sizeof(&p + 1));//
//4/8
printf("%d\n", sizeof(&p[0] + 1));//4/8 , &p[0] + 1得到是'b'的地址
return 0;
}
图解如下:
二维数组
int main()
{
int a[3][4] = { 0 };
printf("%d\n", sizeof(a));//3*4*4 = 48
printf("%d\n", sizeof(a[0][0]));//4
printf("%d\n", sizeof(a[0]));//a[0]是第一行这个一维数组的数组名
//数组名算是单独放在sizeof内部了,计算的是整个数组的大小,大小是16个字节
printf("%d\n", sizeof(a[0] + 1));//?
//a[0]作为第一行的数组名,没有单独放在sizeof内部,没有&
//所以a[0]+1是第一行第二个元素的地址,是地址就是4/8个字节
//
printf("%d\n", sizeof(*(a[0] + 1)));//4
//计算的是就是第一行第2个元素的大小
printf("%d\n", sizeof(a + 1));//4 / 8
//a是数组首元素的地址,是第一行的地址 int(*)[4]
//a+1 就是第二行的地址
printf("%d\n", sizeof(*(a + 1)));//16
//*(a+1) --> a[1] -> sizeof(*(a+1))->sizeof(a[1]) 计算的是第二行的大小
//a+1 --> 是第二行的地址,int(*)[4]
//*(a+1) 访问的第二行的数组
printf("%d\n", sizeof(&a[0] + 1));//4/8
//&a[0]是第一行的地址 int(*)[4]
//&a[0]+1 是第二行的地址 int(*)[4]
printf("%d\n", sizeof(*(&a[0] + 1)));//16 计算的是第二行的大小
printf("%d\n", sizeof(*a));//计算的是第一行的大小-16
//a是数组首元素的地址,就是第一行的地址
//*a 就是第一行
//*a --> *(a+0) --> a[0]//第一行的数组名,表示第一行的地址
printf("%d\n", sizeof(a[3]));//16
//a[3]--> int [4]
//sizeof只关注类型,而不访问内存
return 0;
}
二维数组的图解如下:
在二维数组中:*a=*(a+0)=a[0]
a表示二维数组首元素的地址,即第一行的地址
*a 或(a[0])表示第一行元素的数组名,sizeof(*a)就表示第一行元素的大小,*a 或(a[0])表示二维数组第一行元素的数组名.