【C语言进阶篇】关于指针的八个经典笔试题(图文详解)


在这里插入图片描述

🎬 鸽芷咕:个人主页

 🔥 个人专栏:《C语言初阶篇》 《C语言进阶篇》

⛺️生活的理想,就是为了理想的生活!

文章目录

  • 📋 前言
  • 💬 指针笔试题
    • 💭 笔试题 1:
      • ✅ 代码解析
      • ⁉️ 检验结果:
    • 💭 笔试题 2:
      • ✅ 代码解析
      • ⁉️ 检验结果:
    • 💭 笔试题 3:
      • ✅ 代码解析
      • ⁉️ 检验结果
    • 💭 笔试题 4:
      • ✅ 代码解析
      • ⁉️ 检验结果
    • 💭 笔试题 5:
      • ✅ 代码解析
      • ⁉️ 检验结果
    • 💭 笔试题 6:
      • ✅ 代码解析
      • ⁉️ 检验结果
    • 💭 笔试题 7:
      • ✅ 代码解析
      • ⁉️ 检验结果
    • 💭 笔试题 8:
      • ✅ 代码解析
      • ⁉️ 检验结果
  • 📝全篇总结

📋 前言

  🌈hello! 各位宝子们大家好啊,指针系列的讲解我们就全部讲解完毕了,那么接下来就是来检验成果的时候了?
  ⛳️今天来做一做指针的面试题到底是个什么难度!你真的学会了指针了嘛?这篇文章让你融汇贯通
  📚本期文章收录在《C语言进阶篇》,大家有兴趣可以看看呐
  ⛺️ 欢迎铁汁们 ✔️ 点赞 👍 收藏 ⭐留言 📝!

💬 指针笔试题

💭 笔试题 1:

int main()
{int a[5] = { 1, 2, 3, 4, 5 };int *ptr = (int *)(&a + 1);printf( "%d,%d", *(a + 1), *(ptr - 1));return 0;
}
//程序的结果是什么?

✅ 代码解析


printf( “%d,%d”, *(a + 1), *(ptr - 1));
其实没什么难的,我相信各位铁子们已经想出答案了!关于数组的笔试题我们做了那么多,&数组名就是拿出整个数组的数组名而+1就是跳过整个数组+1;

  • 所以 *(ptr-1)就是回到了数组的最后一个元素的位置

  • 在这里插入图片描述

  • 然后进行解引用 结果就是:5


  • *(a + 1) a 是首元素的地址,所以+1 就是指向第二个元素的地址

  • 在这里插入图片描述

  • 然后进行解引用 结果就是:2

⁉️ 检验结果:

📑图片展示:

在这里插入图片描述

💭 笔试题 2:

//由于还没学习结构体,这里告知结构体的大小是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;
}

🔥 注:本代码是在32位环境下测试的。

✅ 代码解析

printf(“%p\n”, p + 0x1);
这里就非常有意思了指针加一我们都知道是跳过指针类型的大小,整形指针跳过4个字节,字符指针跳过1个指针。而我们这里结构体的大小是20个字节,所以+1跳过的就是20个字节!

  • 但是要注意这里是以16进制输出的所以 20 的16进制就是 14
  • printf(“%p\n”, p + 0x1);
  • 的结果为 0x100014。

printf(“%p\n”, (unsigned long)p + 0x1);
而这里则强制转换成了整形,而强转成整形的话。那么 p 就是一个整数了。而整数+1就是 +1 没有什么变化!

  • printf(“%p\n”, (unsigned long)p + 0x1);
  • 的结果为 0x100001。

printf(“%p\n”, (unsigned int*)p + 0x1);
这里指针 p 被强制转换为了无符号整形,而无符号整形+1。就是跳过4个字节。

  • printf(“%p\n”, (unsigned int*)p + 0x1);
  • 的结果为 0x100004。

🌴 总结:

  • 这里运用了指针的运算和进制位的转换,这要这些知识掌握了。截图就非常easy了!
  • 注:指针运算的博客在这里《指针的运算详解》

⁉️ 检验结果:

在这里插入图片描述

💭 笔试题 3:

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

✅ 代码解析

int *ptr1 = (int *)(&a + 1);
我们先来看一下,&a先把整个数组的地址拿出来然后进行+1然后强制转换为 int 类型的指针传给我们 ptr1。*

  • 那么 *ptr1,不就指向与数组相邻的那个元素了吗!
  • ptr1[-1] 就右回到了,数组的最后一个元素
    在这里插入图片描述

int *ptr2 = (int *)((int)a + 1);
🔥 注:这里需要用到 大小端存储的概念!不会的也去看看 这里只简单讲解哦!

这里就非常有意思了,a是数组首元素的地址。这里却把它强制转换为了整数,然后+1是在括号里面所以a+1,在转换为int* 类型的地址就只跳过了一个字节!

  • 然后再进行+1 跳过4个字节,进行解引用访问4个字节
    在这里插入图片描述

  • 所以int *ptr2 = (int *)((int)a + 1);

  • 🔥 *ptr2 的结果是 0x02000000

这里给大家解释一下大小端存储,就地址存低位,高地址存高位。所以拿出来的时候低地址就是低位,高地址当高位还原的。

  • 这里02是在高地址的所以拿出来的时候直接就放在高位了

⁉️ 检验结果

在这里插入图片描述

💭 笔试题 4:

#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 a[3][2] = { (0, 1), (2, 3), (4, 5) };
这里陷阱就比较多了,一定要注意二维数组的时候是用大括号进行赋值的。而这里使用括号说明他是逗号表达式。

  • 所以a[3][2] = {1,3,5};
  • a[0] = *(a+0) = *p 等于二维数组第一排首元素的地址

在这里插入图片描述

  • int a[3][2] = { (0, 1), (2, 3), (4, 5) };
  • 结果为 1

⁉️ 检验结果

在这里插入图片描述

💭 笔试题 5:

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(*p)[4]; p = a;这里要搞懂赋值给p的是什么,由于 aint (*)[5] 指针 pint (*)[4] 所以会发生隐式转换。到int (*)[4]

在这里插入图片描述

这里我们就可以看到p[4][2]a[4][2]之间相差了,4个整形的长度。

  • 我们知道指针运算里面 指针 - 指针 的=得到的是指针之间元素的个数!
  • printf( “%p,%d\n”, &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]);
  • 所以得到的是 -4 这个数值!

但是如果以地址打印的话得到的就是,-4 在内存中存储的数字。而 -4 在内存中是以补码的形式存储的。
在这里插入图片描述

  • 所以以 %p 打印结果就是 FFFFFFFC。
  • 以 %d 打印的就是 -4 。

⁉️ 检验结果

在这里插入图片描述

💭 笔试题 6:

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

✅ 代码解析

哦豁!这里我们就要对数组足够敏感了。关于数组和数组首页元素的地址。 &aa 取出的是整个二维数组的地址。+1 就是跳过整个数组指向数组后一个元素。

  • **int *ptr1 = (int *)(&aa + 1); **
  • printf( “%d”, *(ptr1 - 1), );
  • 所以在进行 -1 的时候就访问到了数组的最后一个元素 10。

*ptr2 = (int *)(*(aa + 1));这里 aa 代表的就不是首元素的地址了,而是一维数组 aa[0] 整个一维数组的地址,所以+1 就跳到了 aa[1].

  • *ptr2 = (int )((aa + 1));
  • printf( “%d”, *(ptr2 - 1));
  • 所以在进行 -1 就又回到了aa[0] 的最后一个元素 5。

⁉️ 检验结果

在这里插入图片描述

💭 笔试题 7:

#include <stdio.h>
int main()
{char *a[] = {"work","at","alibaba"};char**pa = a;pa++;printf("%s\n", *pa);return 0;
}

✅ 代码解析

这里要注意也是一个陷阱, char *a[] = {“work”,“at”,“alibaba”}; a[ ] 里面可以当成存放的是一维数组字符串首元素的地址。所以当我们把 字符指针 数组 *a[] 存放到 二级指针 pa 里面的时候他们的内存布局就应该是这样的!
在这里插入图片描述

  • pa++ 就指向 a[1] 而对一个地址进行解引用,得到的是a[1] 这个字符指针指向的字符串
  • 所以: printf(“%s\n”, *pa);
  • 的结果是 at

⁉️ 检验结果

在这里插入图片描述

💭 笔试题 8:

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

✅ 代码解析

做这种题首先把图画出来就简单多了
在这里插入图片描述

printf(“%s\n”, **++cpp);
**++cpp指向的就是 cp[1] , 向前了一小步。这时看这我们上面画的图就简单多了!而这时又进行了俩次解引用不就找到 c[2]嘛!
请添加图片描述

  • printf(“%s\n”, **++cpp);
  • 的结果为 POINT。

printf(“%s\n”, *--*++cpp+3);
这个首先要把运算符的逻辑关系搞清楚,做这个题就简单多了 ++ -- 的运算符关系是比 + 号运算优先级要高的。

  • 但是这里要注意前面的 ++ 是会影响指针的指向的 。
  • 那么先++在解引用就找到了这里 *++
    在这里插入图片描述
  • 在 -- 在解引用就找到了这里 *-- , c这个首元素地址
    在这里插入图片描述
  • 但是这里还进行了 +3 那么就要跳过3个 char 字节在打印

在这里插入图片描述

  • printf(“%s\n”, *--*++cpp+3);

  • 所以结果为 ER


printf(“%s\n”, *cpp[-2]+3);
这里我们和上面一样, *cpp[-2]+3 = *(cpp-2)+3 , 我们把它简化一下就简单多了!

  • 这里依旧要注意前面是改变了 cpp的指向 。
  • 所以 cpp-2 然后再进行解引用指向的就是这里
  • 在这里插入图片描述
  • 然后再进行 +3 跳过3个字节进行打印
  • 在这里插入图片描述
  • printf(“%s\n”, *cpp[-2]+3);
  • 结果为 ST

printf(“%s\n”, cpp[-1][-1]+1);
诶这里大家千万不要进入误区了,使用下表引用是不会改变 指针的指向的,所以我们cpp 指向的还是 cp[2]。

  • cpp[-1][-1]+1 = ((cpp-1)-1)+1

  • 先 -1 然后再进行解引用,就找到了这里
    在这里插入图片描述

    • 在 -1 然后再进行解引用,就找到了这里
      在这里插入图片描述
  • 然后再进行 +1 打印出来就是从这个位置开始的!
    在这里插入图片描述

  • printf(“%s\n”, cpp[-1][-1]+1);

  • **结果为 EW **


⁉️ 检验结果

在这里插入图片描述

📝全篇总结

✅ 归纳:
好了以上就是关于经典的八个经典笔试题的详细解析,包含了个个方面的知识不知道大家做的怎么样呢!
  指针 - 指针的运算
  操作符的优先级
  ++ 对指针的影响
  [] *() 的理解
  指针的运算
☁️ 这些就是全部指针 and 数组的全部知识的总结了,不知道大家掌握的怎么样呢?
看到这里了还不给博主扣个:
⛳️ 点赞☀️收藏 ⭐️ 关注

💛 💙 💜 ❤️ 💚💓 💗 💕 💞 💘 💖
拜托拜托这个真的很重要!
你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/322149.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redis (三)

1、redis复制 简单的概括就是主从复制&#xff0c;master以写为主&#xff0c;Slave以读为主&#xff0c;当master数据发生变化的时候&#xff0c;自动将更新的数据异步同步到其他的slave是数据库。 使用这种机制的话&#xff0c;可以做到读写分离&#xff0c;可以减轻主机负担…

【React系列】React生命周期、setState深入理解、 shouldComponentUpdate和PureComponent性能优化、脚手架

本文来自#React系列教程&#xff1a;https://mp.weixin.qq.com/mp/appmsgalbum?__bizMzg5MDAzNzkwNA&actiongetalbum&album_id1566025152667107329) 一. 生命周期 1.1. 认识生命周期 很多的事物都有从创建到销毁的整个过程&#xff0c;这个过程称之为是生命周期&…

uniapp微信小程序投票系统实战 (SpringBoot2+vue3.2+element plus ) -小程序端TabBar搭建

锋哥原创的uniapp微信小程序投票系统实战&#xff1a; uniapp微信小程序投票系统实战课程 (SpringBoot2vue3.2element plus ) ( 火爆连载更新中... )_哔哩哔哩_bilibiliuniapp微信小程序投票系统实战课程 (SpringBoot2vue3.2element plus ) ( 火爆连载更新中... )共计21条视频…

GO语言笔记1-变量与基本数据类型

变量使用步骤 声明赋值使用 package main import "fmt" func main(){var age int //声明一个 int类型的变量叫ageage 18 //给变量用 赋值fmt.Println(age) //使用变量 输出变量的值 } 编译运行输出变量值 变量的四种使用方式 package main import "fmt&q…

【ES6语法学习】解构赋值

文章目录 引言一、什么是解构赋值1.1什么是解构赋值1.2 数组的解构赋值1.2.1 基本用法1.2.2 默认值1.2.3 剩余参数 1.3 对象的解构赋值1.3.1 基本用法1.3.2 默认值1.3.2 剩余参数 1.4 字符串的解构赋值1.5 函数参数的解构赋值 二、解构赋值的优势和应用场景2.1 代码简化和可读性…

2023 IoTDB Summit:清华大学软件学院长聘副教授龙明盛《IoTDB 新组件:内生机器学习》...

12 月 3 日&#xff0c;2023 IoTDB 用户大会在北京成功举行&#xff0c;收获强烈反响。本次峰会汇集了超 20 位大咖嘉宾带来工业互联网行业、技术、应用方向的精彩议题&#xff0c;多位学术泰斗、企业代表、开发者&#xff0c;深度分享了工业物联网时序数据库 IoTDB 的技术创新…

如何借助于AI自研一款换脸app

文章目录 背景涉及的关键技术解析技术流程详解后续待补充 背景 在当今的数字时代&#xff0c;人工智能&#xff08;AI&#xff09;技术已经深入到各个领域&#xff0c;其中之一就是换脸技术。现在&#xff0c;有一个免费的AI换脸应用程序&#xff0c;可以让用户轻松地将自己的…

好的OODA循环与快慢无关

OODA循环是指观察&#xff08;Observe&#xff09;、导向&#xff08;Orient&#xff09;、决策&#xff08;Decide&#xff09;和行动&#xff08;Act&#xff09;这四个步骤的循环过程。它是一种决策和行动的框架&#xff0c;旨在帮助个人或组织更快地适应和应对变化。 OODA循…

Feign远程调用

Feign远程调用 Fegin的使用步骤如下&#xff1a; 1&#xff09;引入依赖 我们在order-service服务的pom文件中引入feign的依赖&#xff1a; <dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign&…

不同阶数的巴特沃斯低通滤波器的空间域表示——数字图像处理

原理 巴特沃斯低通滤波器&#xff08;Butterworth Low-Pass Filter&#xff09;在频率域中的定义是明确的&#xff0c;但它在空间域中的表示不是直观的。这是因为巴特沃斯滤波器的形式是基于频率的&#xff0c;并且其空间域表示涉及到一个复杂的逆傅里叶变换&#xff0c;该变换…

图解算法数据结构-LeetBook-回溯01_机械累加器

请设计一个机械累加器&#xff0c;计算从 1、2… 一直累加到目标数值 target 的总和。注意这是一个只能进行加法操作的程序&#xff0c;不具备乘除、if-else、switch-case、for 循环、while 循环&#xff0c;及条件判断语句等高级功能。 注意&#xff1a;不能用等差数列求和公式…

Leetcode算法系列| 11. 盛最多水的容器

目录 1.题目2.题解C# 解法一&#xff1a;暴力C# 解法二&#xff1a;双指针&#xff08;左指针大于右指针&#xff0c;left&#xff09;C# 解法三&#xff1a;双指针优化&#xff08;左指针小于等于最小高度&#xff0c;left&#xff09;Java 解法一&#xff1a;双指针Python3 解…