【C生万物】 指针和数组笔试题汇总 (上)

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 【C生万物】 指针和数组笔试题汇总 (上)

370e28d82132798f78a90e533d2c4256_e6510fde102248338c249440750db93e.png


前言:

在学习完初级和进级的指针之后,为了检验掌握程度,这里出一期指针和数组的笔试题,看看你掌握的怎么样吧!


目录

Part1:数组笔试题

1.做题

1.1一维数组

1.2字符数组

1.3二维数组

2.总结



Part1:数组笔试题


1.做题


1.1一维数组


计算大小:

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


这里可以暂停,自己做一下。


答案及解析:

int a[] = {1,2,3,4};
printf("%d\n",sizeof(a));        16 -- 整个数组的大小 4*4 == 16
printf("%d\n",sizeof(a+0));      8  -- + 0这个动作表明是指针 64位下指针大小固定为 8
printf("%d\n",sizeof(*a));       4  -- 首元素地址解引用得到首元素1,一个整型大小为4
printf("%d\n",sizeof(a+1));      8  -- 指针大小
printf("%d\n",sizeof(a[1]));     4  -- 元素2的大小
printf("%d\n",sizeof(&a));       8  -- &a代表指向数组a的指针
printf("%d\n",sizeof(*&a));      16 -- 承上,解引用后表示整个数组
printf("%d\n",sizeof(&a+1));     8  -- 指针+1,表示步长,即指针大小
printf("%d\n",sizeof(&a[0]));    8  -- 首元素地址,指针大小
printf("%d\n",sizeof(&a[0]+1));  8  -- 步长,即指针大小


1.2字符数组


计算大小:


第一部分

char arr[] = { 'a','b','c','d','e','f' }; // 没有\0
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));


答案及解析:

char arr[] = { 'a','b','c','d','e','f' };  // 没有\0
printf("%d\n", sizeof(arr));          6 -- 所有字符的总大小,一个字符一字节,共6字节
printf("%d\n", sizeof(arr + 0));      8 -- 表示指针,8字节
printf("%d\n", sizeof(*arr));         1 -- 首元素地址解引用得到首元素,大小为1
printf("%d\n", sizeof(arr[1]));       1 -- 元素b的大小
printf("%d\n", sizeof(&arr));         8 -- 指向整个数组的指针大小
printf("%d\n", sizeof(&arr + 1));     8 -- 指针步长
printf("%d\n", sizeof(&arr[0] + 1));  8 -- 指向首元素指针的步长


第二部分

char arr[] = { 'a','b','c','d','e','f' };  // 没有\0
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[] = { 'a','b','c','d','e','f' };  // 没有\0
printf("%d\n", strlen(arr));         随机值 -- arr表示首元素地址,但不知道\0在哪,不知道停止
printf("%d\n", strlen(arr + 0));     随机值 -- 同上,表示地址
printf("%d\n", strlen(*arr));        报错 -- ==strlen('a'),非法访问
printf("%d\n", strlen(arr[1]));      报错 -- ==strlen('b'),非法访问
printf("%d\n", strlen(&arr));        随机值 -- 从整个数组开始,向后寻找\0停止
printf("%d\n", strlen(&arr + 1));    随机值 -- 从跳过整个数组开始,向后寻找\0停止
printf("%d\n", strlen(&arr[0] + 1)); 随机值 -- 从'b'的地址开始,向后寻找\0停止


第三部分

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


答案及解析:

char arr[] = "abcdef";  // 末尾有\0
printf("%d\n", sizeof(arr));       7 -- 包含\0在内   
printf("%d\n", sizeof(arr+0));     8 -- 指针大小
printf("%d\n", sizeof(*arr));      1 -- a的大小
printf("%d\n", sizeof(arr[1]));    1 -- b的大小
printf("%d\n", sizeof(&arr));      8 -- 字符串指针的大小
printf("%d\n", sizeof(&arr+1));    8 -- 字符串指针的步长大小
printf("%d\n", sizeof(&arr[0]+1)); 8 -- 首元素指针的步长大小


第四部分

char arr[] = "abcdef";  // 末尾有\0
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";  // 末尾有\0
printf("%d\n", strlen(arr));       6 -- 计算\0之前的大小
printf("%d\n", strlen(arr+0));     6 -- 同上
printf("%d\n", strlen(*arr));      报错 -- 给strlen传递'a'
printf("%d\n", strlen(arr[1]));    报错 -- 给strlen传递'b'
printf("%d\n", strlen(&arr));      6 -- 取出起始位置的地址
printf("%d\n", strlen(&arr+1));    随机值 -- 跳过整个数组(包含\0)
printf("%d\n", strlen(&arr[0]+1)); 5 -- 从b开机计算大小,到\0之前


第五部分

char *p = "abcdef";
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));


答案及解析:

char *p = "abcdef";
printf("%d\n", sizeof(p));       8 -- p是指针变量,即指针大小
printf("%d\n", sizeof(p+1));     8 -- p+1是指针变量
printf("%d\n", sizeof(*p));      1 -- 即a的大小
printf("%d\n", sizeof(p[0]));    1 -- 即a的大小
printf("%d\n", sizeof(&p));      8 -- 二级指针大小
printf("%d\n", sizeof(&p+1));    8 -- 指向整个字符串的指针步长
printf("%d\n", sizeof(&p[0]+1)); 8 -- 指向b的指针的大小


第六部分

char *p = "abcdef";
printf("%d\n", sizeof(p));       8 -- p是指针变量,即指针大小
printf("%d\n", sizeof(p+1));     8 -- p+1是指针变量
printf("%d\n", sizeof(*p));      1 -- 即a的大小
printf("%d\n", sizeof(p[0]));    1 -- 即a的大小
printf("%d\n", sizeof(&p));      8 -- 二级指针大小
printf("%d\n", sizeof(&p+1));    8 -- 指向整个字符串的指针步长
printf("%d\n", sizeof(&p[0]+1)); 8 -- 指向b的指针的大小


答案及解析:

char* p = "abcdef";                
printf("%d\n", strlen(p));         6 -- 标准求字符串长度
printf("%d\n", strlen(p + 1));     5 -- 从b开始求字符串长度
printf("%d\n", strlen(*p));        报错 -- strlen(a)
printf("%d\n", strlen(p[0]));      报错 -- 同上
printf("%d\n", strlen(&p));        随机值 -- 指向p的指针,另起
printf("%d\n", strlen(&p + 1));    随机值 -- 跳过指向p的指针
printf("%d\n", strlen(&p[0] + 1)); 5 -- 从b开始求字符串长度


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


答案及解析:

因为是二维数组,这里画个图会好理解些:

bba226494a3ad6f84aaebb7ca36b20c1_e0d4f782ed654932969bc8367555d389.png

注意:可以先这样理解,实际内存中是连续存储的。

int a[3][4] = {0};                 
printf("%d\n",sizeof(a));           48 -- 整个二位数组的大小 3*4*4
printf("%d\n",sizeof(a[0][0]));      4 -- 下标为0,0,即第一个元素的大小
printf("%d\n",sizeof(a[0]));        16 -- a[0]作为第一行的数组名,单独放在sizeof内部,计算的是数组第一行的大小
printf("%d\n",sizeof(a[0]+1));       8 --a[0]作为第一行的数组名,没有单独放在sizeof内部,没有取地址,表示的就是数组首元素的地址,即a[0][0]的地址,再+1就是a[0][1]的地址,是地址就是指针,指针大小
printf("%d\n",sizeof(*(a[0]+1)));    4 -- 第一行第二个元素的大小
printf("%d\n",sizeof(a+1));          8 -- a是二维数组的数组名,数组名表示首元素的地址,就是第一行的地址,a+1就是第二行的地址,是地址就是指针,指针大小
printf("%d\n",sizeof(*(a+1)));       16 -- 第二行的大小
printf("%d\n",sizeof(&a[0]+1));       8 -- &a[0]是第一行的地址,&a[0]+1就是第二行的地址,地址的大小指针大小
printf("%d\n",sizeof(*(&a[0]+1)));   16 -- 第二行的大小
printf("%d\n",sizeof(*a));           16 -- a表示首元素的地址,就是第一行的地址,*a就是第一行,计算的就是第一行的大小
printf("%d\n",sizeof(a[3]));         16 -- 如果数组存在第四行,a[3]就是第四行的数组名,数组名单独放在sizeof内部,计算的是第四行的大小


2.总结

数组名的意义:

• sizeof( 数组名 ) ,这里的数组名表示整个数组,计算的是整个数组的大小;

• &数组名,这里的数组名表示整个数组,取出的是整个数组的地址;

• 除此之外所有的数组名都表示首元素的地址。


总结:

这篇博客带大家做了些关于数组和指针的笔试题,基本上汇总了全部类型,其实到这里还没完,那么下篇再带大家继续指针的练习。


码文不易

如果你觉得这篇文章还不错并且对你有帮助,不妨支持一波哦  💗💗💗


目录
相关文章
|
2月前
使用指针访问数组元素
【10月更文挑战第30天】使用指针访问数组元素。
41 3
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
2月前
|
容器
在使用指针数组进行动态内存分配时,如何避免内存泄漏
在使用指针数组进行动态内存分配时,避免内存泄漏的关键在于确保每个分配的内存块都能被正确释放。具体做法包括:1. 分配后立即检查是否成功;2. 使用完成后及时释放内存;3. 避免重复释放同一内存地址;4. 尽量使用智能指针或容器类管理内存。
|
2月前
|
存储 NoSQL 编译器
C 语言中指针数组与数组指针的辨析与应用
在C语言中,指针数组和数组指针是两个容易混淆但用途不同的概念。指针数组是一个数组,其元素是指针类型;而数组指针是指向数组的指针。两者在声明、使用及内存布局上各有特点,正确理解它们有助于更高效地编程。
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
61 4
|
2月前
使用指针访问数组元素
【10月更文挑战第31天】使用指针访问数组元素。
53 2
|
2月前
|
算法 索引
单链表题+数组题(快慢指针和左右指针)
单链表题+数组题(快慢指针和左右指针)
41 1
|
3月前
|
存储
如何使用指针数组来实现动态二维数组
指针数组可以用来实现动态二维数组。首先,定义一个指向指针的指针变量,并使用 `malloc` 为它分配内存,然后为每个子数组分配内存。通过这种方式,可以灵活地创建和管理不同大小的二维数组。
|
3月前
|
存储
如何通过指针数组来实现二维数组?
介绍了二维数组和指针数组的概念及其区别,详细讲解了如何使用指针数组模拟二维数组,包括定义与分配内存、访问和赋值元素、以及正确释放内存的步骤,适用于需要动态处理二维数据的场景。