进阶C语言指针和数组笔试题解析

Posted 热爱跑步的恒川

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了进阶C语言指针和数组笔试题解析相关的知识,希望对你有一定的参考价值。

指针和数组笔试题解析

1. 一维数组

1.1 回顾知识点

  1. 数组和指针
    数组 - 能够存放一组相同类型的元素,数组的大小取决于数组的元素个数和元素类型
    指针 - 地址/指针变量 ,大小是4/8个字节
    数组是数组,指针是指针,二者不等价
    数组名是数组首元素的地址,这个地址就可以存放在指针变量中,我们就可以使用指针来遍历数组
  2. 数组名
    大部分情况下数组名是数组首元素的地址
    但是有2个例外:
    sizeof(数组名) - 数组名表示整个数组,计算的是整个数组的大小
    &数组名 - 数组名表示整个数组,取出的是数组的地址

1.2 思考该结果是什么?

int main()

	int a[] =  1,2,3,4 ;
	printf("%d\\n", sizeof(a));
	printf("%d\\n", sizeof(a + 0));
	printf("%d\\n", sizeof(*a));
	printf("%d\\n", sizeof(a + 1));
	printf("%d\\n", sizeof(a[1]));
	printf("%d\\n", sizeof(&a));
	printf("%d\\n", sizeof(*&a));
	printf("%d\\n", sizeof(&a + 1));
	printf("%d\\n", sizeof(&a[0]));
	printf("%d\\n", sizeof(&a[0] + 1));
	return 0;

详细讲解:

int main()

	int a[] =  1,2,3,4 ;
	printf("%d\\n", sizeof(a));//16
	//sizeof(a)就是数组名单独放在sizeof内部,计算的数组总大小,单位是字节
	printf("%d\\n", sizeof(a + 0));//4/8 个字节
	//a+0 其实是数组首元素的地址
	printf("%d\\n", sizeof(*a));//4  它是一个元素的大小并不是一个地址的大小,所以不可能是4/8个字节
	//a是数组首元素的地址 - &a[0]
	//*a -> *&a[0] -> a[0]
	printf("%d\\n", sizeof(a + 1));//4/8
	//a是数组首元素的地址 -- int*
	//a+1 跳过1个整型, 是第二个元素的地址
	//
	printf("%d\\n", sizeof(a[1]));//4
	printf("%d\\n", sizeof(&a));//4/8
	//&a - 取出的是数组的地址,但是数组的地址也是地址呀,是地址大小就是4/8字节
	//int (*pa)[4] = &a;//int(*)[4]
	//
	printf("%d\\n", sizeof(*&a));//16
	//sizeof(a)
	//int(*)[4]数组指针
	//
	printf("%d\\n", sizeof(&a + 1));//4/8
	//&a -->  int (*)[4]
	//&a+1 跳过一个数组

	printf("%d\\n", sizeof(&a[0]));//取出首元素的地址 4/8
	printf("%d\\n", sizeof(&a[0] + 1));//第二个元素的地址

	return 0;

图片讲解:

2. 字符数组

2.1 回顾知识点

  1. sizeof 计算的是占用内存空间的大小,单位是字节,不关注内存中到底存放的是什么。
  2. sizeof 不是函数,是操作符。
  3. strlen 是函数。
  4. strlen 是针对字符串的,求的是字符串的长度,本质上统计的是\\0之前出现的字符个数。

2.2 思考该结果是什么?

int main()

	char arr[] =  'a','b','c','d','e','f' ;
	printf("%d\\n", sizeof(arr));
	printf("%d\\n", sizeof(arr + 0));
	printf("%d\\n", sizeof(*arr));
	printf("%d\\n", sizeof(arr[1]));
	printf("%d\\n", sizeof(&arr));
	printf("%d\\n", sizeof(&arr + 1));
	printf("%d\\n", sizeof(&arr[0] + 1));
	printf("%d\\n", strlen(arr));
	printf("%d\\n", strlen(arr + 0));
	printf("%d\\n", strlen(*arr));
	printf("%d\\n", strlen(arr[1]));
	printf("%d\\n", strlen(&arr));
	printf("%d\\n", strlen(&arr + 1));
	printf("%d\\n", strlen(&arr[0] + 1));
	return 0;

详细讲解:

int main()

	char arr[] =  'a','b','c','d','e','f' ;
	printf("%d\\n", sizeof(arr));//6
	printf("%d\\n", sizeof(arr + 0));//arr+0是数组首元素的地址 4/8
	printf("%d\\n", sizeof(*arr));//*arr是首元素的,计算的是首元素的大小 1
	printf("%d\\n", sizeof(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));//4/8 第二个元素的地址
	return 0;

图片讲解:

详细讲解:

int main()

	char arr[] =  'a','b','c','d','e','f' ;
	
	printf("%d\\n", strlen(arr));//随机值,因为不知道\\0的位置
	printf("%d\\n", strlen(arr + 0));//随机值
	printf("%d\\n", strlen(*arr));//非法访问  我需要的是一个地址,你却给了我一个元素,强行访问。
	printf("%d\\n", strlen(arr[1]));//'b' - 98 当成地址,形参非法访问
	printf("%d\\n", strlen(&arr));//随机值
	printf("%d\\n", strlen(&arr + 1));//随机值-6
	printf("%d\\n", strlen(&arr[0] + 1));//随机值-1

图片讲解:

形成非法访问的原因:

我需要的是一个地址,你却给了我一个元素,强行访问,把a的ASCII值当成了地址。

2.3 思考另一组

int main()

	char arr[] = "abcdef";
	printf("%d\\n", sizeof(arr));
	printf("%d\\n", sizeof(arr + 0));
	printf("%d\\n", sizeof(*arr));
	printf("%d\\n", sizeof(arr[1]));
	printf("%d\\n", sizeof(&arr));
	printf("%d\\n", sizeof(&arr + 1));
	printf("%d\\n", sizeof(&arr[0] + 1));
	printf("%d\\n", strlen(arr));
	printf("%d\\n", strlen(arr + 0));
	printf("%d\\n", strlen(*arr));
	printf("%d\\n", strlen(arr[1]));
	printf("%d\\n", strlen(&arr));
	printf("%d\\n", strlen(&arr + 1));
	printf("%d\\n", strlen(&arr[0] + 1));
	return 0;

详细讲解:
sizeof 部分:

int main()

	char arr[] = "abcdef";//[a b c d e f \\0]

	printf("%d\\n", sizeof(arr));//7
	printf("%d\\n", sizeof(arr + 0));//4/8
	printf("%d\\n", sizeof(*arr));//*arr -是数组首元素 1
	//arr[0]  =  *(arr+0) = *(&arr[0])
	//int sz = sizeof(arr)/sizeof(*arr);
	//int sz = sizeof(arr)/sizeof(arr[0]);两种表达形式一样

	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

	return 0;

strlen 部分:

int main()

	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
	//&arr - char(*)[7]  数组指针
	printf("%d\\n", strlen(&arr + 1));//随机值
	printf("%d\\n", strlen(&arr[0] + 1));//5

	return 0;

图片讲解:

2.4 再来一组对比练习

int main()

	char* p = "abcdef";  //p里面放的是a的地址
	printf("%d\\n", sizeof(p));
	printf("%d\\n", sizeof(p + 1));
	printf("%d\\n", sizeof(*p));
	printf("%d\\n", sizeof(p[0]));
	printf("%d\\n", sizeof(&p));
	printf("%d\\n", sizeof(&p + 1));
	printf("%d\\n", sizeof(&p[0] + 1));
	printf("%d\\n", strlen(p));
	printf("%d\\n", strlen(p + 1));
	printf("%d\\n", strlen(*p));
	printf("%d\\n", strlen(p[0]));
	printf("%d\\n", strlen(&p));
	printf("%d\\n", strlen(&p + 1));
	printf("%d\\n", strlen(&p[0] + 1));
	return 0;

详细讲解:
sizeof 部分:

int main()

    char* p = "abcdef";
	printf("%d\\n", sizeof(p));//4 / 8  p里面放的是a的地址
	printf("%d\\n", sizeof(p + 1));//'b'的地址,4/8
	printf("%d\\n", sizeof(*p));//1
	printf("%d\\n", sizeof(p[0]));//*(p+0)--'a' 1
	printf("%d\\n", sizeof(&p));//4 /8
	printf("%d\\n", sizeof(&p + 1));//4/8
	printf("%d\\n", sizeof(&p[0] + 1));//&p[0]+1是'b'的地址 4/8

	return 0;

图片讲解:


strlen 部分:

int main()

	char* p = "abcdef";
	printf("%d\\n", strlen(p));//6
	printf("%d\\n", strlen(p + 1));//p+1是'b'的地址 5
	printf("%d\\n", strlen(*p));//err  //把a的ASCII值当成了地址
	printf("%d\\n", strlen(p[0]));//err
	printf("%d\\n", strlen(&p));//随机值
	printf("%d\\n", strlen(

让学指针变得更简单

文章目录


前言

指针面试题,对指针的理解不再停留在简单的知识层面上,而是可以知道面试题中指针的考察是怎样的;


一、指针和数组笔试题解析

1.1 一维数组

首先说一下知识点:很重要!!!
数组名的意义:

  1. sizeof(数组名),这里的数组名表示整个数组,计算的是整个数组的大小。
  2. &数组名,这里的数组名表示整个数组,取出的是整个数组的地址。
  3. 除此之外所有的数组名都表示首元素的地址。

int a[ ] = 1,2,3,4;
printf( “%d\\n”,sizeof(a) );
printf( “%d\\n”,sizeof(a+0) );
printf( “%d\\n”,sizeof(a) );
printf( “%d\\n”,sizeof(a+1) );
printf( “%d\\n”,sizeof(a[1]) );
printf( “%d\\n”,sizeof(&a) );
printf( “%d\\n”,sizeof(
&a) ;
printf( “%d\\n”,sizeof(&a+1) );
printf( “%d\\n”,sizeof(&a[0]) );
printf( “%d\\n”,sizeof(&a[0]+1) );


1.2 字符数组

char arr[] = ‘a’,‘b’,‘c’,‘d’,‘e’,‘f’;
printf(“%d\\n”, sizeof(arr));
printf(“%d\\n”, sizeof(arr+0));
printf(“%d\\n”, sizeof(*arr));
printf(“%d\\n”, sizeof(arr[1]));
printf(“%d\\n”, sizeof(&arr));
printf(“%d\\n”, sizeof(&arr+1));
printf(“%d\\n”, sizeof(&arr[0]+1));
printf(“%d\\n”, strlen(arr));
printf(“%d\\n”, strlen(arr+0));
printf(“%d\\n”, strlen(*arr));
printf(“%d\\n”, strlen(arr[1]));
printf(“%d\\n”, strlen(&arr));
printf(“%d\\n”, strlen(&arr+1));
printf(“%d\\n”, strlen(&arr[0]+1));




char arr[] = “abcdef”;
printf(“%d\\n”, sizeof(arr));
printf(“%d\\n”, sizeof(arr+0));
printf(“%d\\n”, sizeof(*arr));
printf(“%d\\n”, sizeof(arr[1]));
printf(“%d\\n”, sizeof(&arr));
printf(“%d\\n”, sizeof(&arr+1));
printf(“%d\\n”, sizeof(&arr[0]+1));
printf(“%d\\n”, strlen(arr));
printf(“%d\\n”, strlen(arr+0));
printf(“%d\\n”, strlen(*arr));
printf(“%d\\n”, strlen(arr[1]));
printf(“%d\\n”, strlen(&arr));
printf(“%d\\n”, strlen(&arr+1));
printf(“%d\\n”, strlen(&arr[0]+1));




char *p = “abcdef”;
printf(“%d\\n”, sizeof§);
printf(“%d\\n”, sizeof(p+1));
printf(“%d\\n”, sizeof(*p));
printf(“%d\\n”, sizeof(p[0]));
printf(“%d\\n”, sizeof(&p));
printf(“%d\\n”, sizeof(&p+1));
printf(“%d\\n”, sizeof(&p[0]+1));
printf(“%d\\n”, strlen§);
printf(“%d\\n”, strlen(p+1));
printf(“%d\\n”, strlen(*p));
printf(“%d\\n”, strlen(p[0]));
printf(“%d\\n”, strlen(&p));
printf(“%d\\n”, strlen(&p+1));
printf(“%d\\n”, strlen(&p[0]+1));



1.3 二维数组

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]) );


2. 指针笔试题

int main()

int a[5] = 1, 2, 3, 4, 5 ;
int *ptr = (int *)(&a + 1);
printf( “%d,%d”, *(a + 1), *(ptr - 1));
return 0;

//程序的结果是什么?


//由于还没学习结构体,这里告知结构体的大小是20个字节
struct Test

int Num;
char *pcName;
short sDate;
char cha[2];
short sBa[4];
*p;
//假设p 的值为0x100000。 如下表表达式的值分别为多少?
//已知,结构体Test类型的变量大小是20个字节
int main()

printf(“%p\\n”, p + 0x1);
printf(“%p\\n”, ( unsigned long )p + 0x1);
printf(“%p\\n”, (unsigned int * )p + 0x1);
return 0;


int main()

int a[4] = 1, 2, 3, 4 ;
int *ptr1 = (int *)(&a + 1);
int *ptr2 = (int *)((int)a + 1);
printf( “%x,%x”, ptr1[-1], *ptr2);
return 0;


#include <stdio.h>
int main()

int a[3][2] = (0, 1), (2, 3), (4, 5) ;
int *p;
p = a[0];
printf( “%d”, p[0]);
return 0;


int main()

int a[5][5];
int(*p)[4];
p = a;
printf( “%p,%d\\n”, &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
return 0;


int main()

int aa[2][5] = 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 ;
int *ptr1 = (int *)(&aa + 1);
int *ptr2 = (int * )( * (aa + 1) );
printf( “%d,%d”, * (ptr1 - 1), *(ptr2 - 1) );
return 0;


#include <stdio.h>
int main()

char *a[] = “work”,“at”,“alibaba”;
char**pa = a;
pa++;
printf(“%s\\n”, *pa);
return 0;


int main()

char * c[ ] = “ENTER”,“NEW”,“POINT”,“FIRST”;
char ** cp[] = c+3,c+2,c+1,c;
char *** cpp = cp;
printf(“%s\\n”, ** ++cpp);
printf(“%s\\n”, * --* ++cpp+3);
printf(“%s\\n”, * cpp[-2]+3);
printf(“%s\\n”, cpp[-1][-1]+1);
return 0;



总结

指针的博客就告一段落了,敬请期待后面的博客吧!!!

以上是关于进阶C语言指针和数组笔试题解析的主要内容,如果未能解决你的问题,请参考以下文章

C语言篇 + 指针进阶练习 + qsort模拟实现(回调函数思想) + 指针和数组笔试题

指针进阶—指针和数组笔试题解析[建议收藏]

指针进阶—指针和数组笔试题解析[建议收藏]

C语言进阶之旅 (11.5)指针下 提升篇

C语言进阶之旅(留下的足迹)

C语言进阶之旅(留下的足迹)