【C语言】——指针八:指针运算笔试题解析

【C语言】——指针八:指针运算笔试题解析

    • 一、题一
    • 二、题二
    • 三、题三
    • 四、题四
    • 五、题五
    • 六、题六
    • 七、题七

一、题一

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

  这道题 * (a + 1) 相信大家都没问题,它表示的是数组的第二个元素
  
我们来重点讲一下*(ptr - 1)

  • 先来看 &a。对数组名进行取地址操作, 取出的是整个数组的地址,虽然数值上与数组首元素相等,但他们是不同的类型。在这里,它类型为 i n t ( ∗ ) [ 5 ] int(*)[5] int[5]
      
  • 再看 &a + 1 ,这里取出的是整个数组的地址,+1 即跳过整个数组。最后,再强制类型转换成 i n t int int * 类型
      
  • ptr - 1 此时,指针已经跳过了整个数组,因为此时的 p t r ptr ptr 已经是 i n t int int* 类型, - 1,后退 4 个字节,即指向数组最后一个元素
      
  • *(ptr - 1) 最后再进行解引用,取出数组最后一个元素

在这里插入图片描述

运行结果:

图:

  
  

  
  
  

二、题二

//在x86环境下
//假设结构体大小20字节
//程序输出结果是啥
struct Test
{int Num;char* pcName;short sDate;char cha[3];short sBa[4];
}*p = (struct Test*)0x100000;int main()
{printf("%p\n", p + 0x1);printf("%p\n", (unsigned long)p + 0x1);printf("%p\n", (unsigned int*)p + 0x1);return 0;
}

  

  • 首先我们来理解(定义的结构体类型)*p = (struct Test*)0x100000; 什么意思呢?即定义了一个结构体类型,创建了该类型的指针变量 p p p 。同时,将0x100000这个十六进制数强制类型转换成该结构体指针类型(整数默认 i n t int int 类型)后,将其赋值指针变量 p p p
      
  • printf("%p\n", p + 0x1); 指针类型 +1,跳过的是该指针所指向元素的类型的大小个字节。这里定义的结构体大小为 20 字节,即跳过 20 字节,因为是十六进制表示,即:0x100000 + 0x14,答案:0x100014
      
  • printf("%p\n", (unsigned long)p + 0x1); 第二个 p r i n t f printf printf 语句,先将 p p p 强制类型转换成 u n s i g n e d unsigned unsigned l o n g long long 类型,此时0x100000 仅仅是 一个普通的数字而已,再加 0x1,是单纯的数学上的相加。答案:0x100001
      
  • printf("%p\n", (unsigned int*)p + 0x1); 第三个 p r i n t f printf printf 语句,先将 p p p 强制类型转换成 u n s i g n e d unsigned unsigned i n t int int *类型, u n s i g n e d unsigned unsigned i n t int int 大小为 4 个字节,所以 +1 跳过 4 个字节,十六进制表示。答案:0x100004

  
运行结果:
图

  
  
  
  
  

三、题三

int main()
{int a[3][2] = { (0,1),(2,3),(4,5) };int* p;p = a[0];printf("%d\n", p[0]);return 0;
}

  

  • int a[3][2] = { (0,1),(2,3),(4,5) };先来看对数组的初始化。这里有个小坑点 { (0,1 ), (2,3 ), (4,5 ) }大括号内是三个小括号,表示的是逗号表达式,并不是三个大括号,按行初始化。逗号表达式,从左到右一次执行,整个表达式的结果是最后一个表达式的结果

  因此数组中存放的数据是:

在这里插入图片描述

  

  • p = a[0]; a [ 0 ] 可以看做是二维数组中,第一行数组的数组名,数组名表示的是数组首元素的地址,即 p p p 中存放的是 1 的地址
      
  • printf("%d\n", p[0]); p [ 0 ] 相当于*(p + 0) == *p,即打印 1
      

运行结果:
图:

  
  
  
  
  

四、题四

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;
}
  • 理解这题,关键的是理解&p[4][2]&a[4][2]分别指向哪个元素。
      
  • 首先来看&a[4][2]
    • & a [ 4 ] [ 2 ] a[4][2] a[4][2] 等价于 & * ( * (a + 4)+ 2),& 和 * 相互抵消,即等价于 * (a + 4)+ 2。这里 a a a 是什么呢? a a a 是第一行这个一维数组的地址,类型为 i n t ( ∗ ) [ 5 ] int(*)[5] int()[5] a a a + 4,指向第五行。再对其进行解引用,得到的是第五行这个数组,即第五行的数组名,即第五行数组首元素的地址,对其 +2,指向第五行第三个元素

  

图:

  

  • 接着我们来看&p[4][2]
    • p p p 的类型为 i n t ( ∗ ) [ 4 ] int(* )[4] int()[4] ,题目中将 a a a 的地址给 p p p ,其实会报警告,因为 a a a i n t ( ∗ ) [ 5 ] int( * )[5] int()[5] 类型,类型不匹配,但程序能运行。
    • &p[4][2] 等价于* (p + 4)+ 2。首先,我们要理解 p p p + 1 跳过几个字节。因为 p p p 的类型为 i n t ( ∗ ) [ 4 ] int(* )[4] int()[4] ,所以在 p p p 眼里,二维数组 a a a 是一个每行有 4 个元素的数组+1 跳过的是 4 个整型,即 16 个字节。而 p p p + 4 则是过了 16 个整型,即 64 个字节
        
        
  • * (p + 4) + 2:对 p p p 进行解引用,再 +2,这个过程与 * (a + 4)+ 2 相同,即指针移动了两个整型大小的字节,指向第三个元素
      

-

  

  • 可以看到,&p[4][2]&a[4][2]之间相差了 4 个元素,因为&p[4][2]是小地址,所以 &p[4][2] - &a[4][2] = -4
      

  • printf("%p,%d\n", &p[4][2] - &a[4][2], &p[4][2] - &a[4][2]); &p[4][2] - &a[4][2]的结果是 - 4,因此以%d的形式打印没问题。那以%p形式打印呢?%p是打印地址,地址肯定是无符号类型。- 4 的补码是11111111 11111111 11111111 11111100,转为十六进制为 FF FF FF FC
      

运行结果:

在这里插入图片描述

  
  
  
  
  

五、题五

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\n", *(ptr1 - 1), *(ptr2 - 1));return 0;
}

  

  • 这道题的数组初始化没有坑啊,不用担心。
      
  • int* ptr1 = (int*)(&aa + 1); & a a aa aa:& + 数组名,取出的是整个数组的地址,+1 跳过整个数组,再将其强制类型转换成 i n t int int* 类型。
      
  • int* ptr2 = (int*)(*(aa + 1)); a a aa aa数组首元素的地址,即第一行的地址,+1 跳过第一行,指向第二行。对其进行解引用,得到的是第二行的数组名,即第二行首元素的地址。其实这里强制类型转换成 i n t int int *是多余的,第二行首元素的地址本来就是 i n t int int *类型。
      
  • printf("%d %d", *(ptr1 - 1), *(ptr2 - 1)); 此时 * ( p t r 1 − 1 ) (ptr1 - 1) (ptr11) 与 * ( p t r 2 − 1 ) (ptr2 - 1) (ptr21) 都是 i n t int int *类型,- 1 后退 4 个字节,即一个整型元素。
      

图:

  
运行结果:
在这里插入图片描述

  
  
  
  
  

六、题六

int main()
{char* a[] = { "work","at","alibaba" };char** pa = a;pa++;printf("%s\n", *pa);return 0;
}

  

  • char* a[ ] = { "work","at","alibaba" }; 这是一个指针数组,数组中每个元素都放着指向对应字符串的首元素的地址。
      
  • char** pa = a; 这里取出数组首元素的地址,因为数组首元素原本类型为指针类型,所以这里 p a pa pa二级指针
      
  • pa++; p a pa pa = p a pa pa + 1,指向数组第二个元素的地址。
      
  • printf("%s\n", *pa); p a pa pa 解引用,得到指向 “ a t at at” 的地址,打印 a t at at
      

运行结果:
图:
  
  可能有些小伙伴对二级指针有点迷糊,这张图就一目了然啦(这里是 x86 环境下,x64 环境下指针变量大小 8 字节)
  

在这里插入图片描述

  
  这里我们可以把各个地址打印出来观察

int main()
{char* a[] = { "work","at","alibaba" };printf("%p\n", &a[0]);printf("%p\n", &a[1]);printf("%p\n", &a[2]);printf("\n");printf("%p\n", a[0]);printf("%p\n", a[1]);printf("%p\n", a[2]);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;
}

  
  要理解这道题,画图是必不可少的,我们先把图画出来

在这里插入图片描述

  

printf("%s\n", **++cpp);
  

  • 前置++ 与解引用操作符 ∗ * 的优先级一致,根据结合性从右往左运算
  • 先是 c p p cpp cpp 自增,自增后指向 c p [ 1 ] cp[1] cp[1]
  • 第一次解引用,找到 c p [ 1 ] cp[1] cp[1],第二次解引用找到 c [ 2 ] c[2] c[2]
  • 最终打印:POINT
      
      
    在这里插入图片描述

  

printf("%s\n", *-- * ++cpp + 3);
  

  • 首先 + 运算符优先级最小,因此 +3 最后才算
  • 其他四个操作符优先级一样,根据结合性从左到右计算
  • 先是 c p p cpp cpp 自增,自增后指向 c p [ 2 ] cp[2] cp[2]
  • c p p cpp cpp 解引用,找到 c p [ 2 ] cp[2] cp[2]
  • c p [ 2 ] cp[2] cp[2] 自减, c c c + 1 自减为 c c c ,原本指向 c [ 1 ] c[1] c[1],自减后指向 c [ 0 ] c[0] c[0]
  • c p [ 2 ] cp[2] cp[2] 解引用,找到 c [ 0 ] c[0] c[0]
  • c [ 0 ] c[0] c[0] 指向的是 “ENTER” 字符串的首元素 'E’+3 则指向 ‘E’
  • 答案:ER

  
  
在这里插入图片描述

  

printf("%s\n", *cpp[-2] + 3);
  

  • 首先来看 c p p [ − 2 ] cpp[-2] cpp[2],它等价于 * ( c p p − 2 ) (cpp - 2) (cpp2) c p p cpp cpp - 2 后再解引用(注: c p p cpp cpp 本身值没变),找到 c p [ 0 ] cp[0] cp[0]
  • 接着,再解引用,找到 c [ 3 ] c[3] c[3]
  • c [ 3 ] c[3] c[3] 指向 “FIRST” 字符串的首元素 'F’+3 则指向 ‘S’
  • 答案:ST

  在这里插入图片描述

  

printf("%s\n", cpp[-1][-1] + 1);
  

  • c p p [ − 1 ] [ − 1 ] cpp[-1][-1] cpp[1][1] 等价于 * ( * ( c p p cpp cpp - 1 ) - 1),先来看里面的 *( c p p cpp cpp - 1),将其解引用,找到 c p [ 1 ] cp[1] cp[1]
  • 接着,再看 * ( c p [ 1 ] − 1 ) (cp[1] - 1) (cp[1]1) c p [ 1 ] − 1 cp[1] - 1 cp[1]1,原本指向 c [ 2 ] c[2] c[2]- 1 后指向 c [ 1 ] c[1] c[1]
  • c [ 1 ] c[1] c[1] 指向 “NEW” 字符串的首元素 'N’+1 则指向 ‘E’
  • 答案:EW

  
  在这里插入图片描述

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

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

相关文章

揭开“栈和队列”的神秘面纱

前言 在线性表中不止有顺序表和链表,今天的主角就如标题所说--->认识栈和队列。把他们俩放一起总结是有原因的,还请看官听我娓娓道来~ 什么是栈? 栈(stack)是限定仅在表尾进行插入和删除操作的线性表 咱可以把栈理…

Pro版 v3.0首页DIY热区神器,让图片更全能!

Pro版v3.0新增了很多新功能,为了方便大家能快速了解使用,今天,我们就开始来逐步了解这些新功能。 在此次更新中,Pro版的首页DIY功能实现了全新重构升级,新增了DIY热区、视频、排行榜、积分商城、预售、签到组件&#…

【Hadoop技术框架-MapReduce和Yarn的详细描述和部署】

前言: 💞💞大家好,我是书生♡,今天的内容主要是Hadoop的后两个组件:MapReduce和yarn的相关内容。同时还有Hadoop的完整流程。希望对大家有所帮助。感谢大家关注点赞。 💞💞前路漫漫&…

深度学习实战73-基于多模态CLIP模型的实战项目,CLIP模型的架构介绍与代码实现

大家好,我是微学AI,今天给大家介绍一下深度学习实战73-基于多模态CLIP模型的实战项目,CLIP模型的架构介绍与代码实现。多模态CLIP(Contrastive Language-Image Pre-training)模型是一种深度学习模型,其核心设计理念是通过大规模的对比学习训练,实现图像与文本之间的跨模…

【JAVASE】面向对象程序三大特性之一( 封装)

✅作者简介:大家好,我是橘橙黄又青,一个想要与大家共同进步的男人😉😉\n 🍎个人主页:再无B~U~G-CSDN博客 目标: 1.包的使用 2.static关键字的使用 3.代码…

Python+Vuecil笔记

Nginx 进入目录: C:\nginx-1.20.2\nginx-1.20.2 start nginx 开始 nginx -s stop 停止 nginx -s quit 退出CSS 通过标签去写css 循环展示数据 JS 点击时执行事件 Django 配置media 在seetings里面修改 STATIC_URL /static/ MEDIA_URL /upload/ MEDIA_ROOT os.pat…

windows terminal美化教程

安装terminal 微软商店下载安装terminal 配置文件 进入terminal,打开设置。 {"$schema": "https://aka.ms/terminal-profiles-schema",// global settings"profiles": {// profile settings"defaults": {// default sett…

阿里云服务器租赁一年收费标准

阿里云服务器租用价格表2024年最新,云服务器ECS经济型e实例2核2G、3M固定带宽99元一年,轻量应用服务器2核2G3M带宽轻量服务器一年61元,ECS u1服务器2核4G5M固定带宽199元一年,2核4G4M带宽轻量服务器一年165元12个月,2核…

【亲测有效】微信公众号设置菜单栏显示,未开启自定义菜单,微信公众平台自定义菜单接口开发

微信公众平台自定义菜单接口开发 问题:运营人员在设置微信公众号设置菜单栏显示,未开启自定义菜单解决方案(微信公众平台自定义菜单接口开发):自定义菜单-创建接口请求链接完整代码第一步:在WeChat类里添加代码情况一:没有WeChat类情况,如果已有请看情况二情况二:已有…

【系统架构师】-软件架构设计

1、软件架构的概念 架构的本质 1、软件架构为软件系统提供了一个结构、行为和属性的高级抽象。 2、软件架构风格是特定应用领域的惯用模式,架构定义一个词汇表和一组约束。 架构的作用 1、软件架构是项目干系人进行交流的手段。 2、软件架构是可传递和可复用的模型…

番茄 短abogus补环境

声明: 本文章中所有内容仅供学习交流使用,不用于其他任何目的,抓包内容、敏感网址、数据接口等均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关!wx a15018601872 本文章…

端到端单倍型参考基因组揭示了三倍体香芽蕉型香蕉亚基因组的分歧和疾病抵抗力-文献精读-5

T2T基因组文献分享Telomere-to-telomere haplotype-resolved reference genome reveals subgenome divergence and disease resistance in triploid Cavendish banana 三倍体植物基因组的文献,各位同仁还有什么有特色的基因组评论区留言~ 摘要 香蕉是世界上最重要…

JavaEE——手把手教你实现简单的 servlet 项目

文章目录 一、什么是 Servlet二、创建一个简单的 Servlet 程序1. 创建项目2.引入依赖3. 创建目录4.编写代码5. 打包程序6. 部署7.验证整体过程总结 三、使用 Smart Tomcat 插件简化项目创建四、创建项目时可能遇到的几个问题。 一、什么是 Servlet Servlet 是一种实现 动态页面…

12、最小覆盖子串

如何想到这个解法 问题的特点: 首先,认识到这是一个关于子串的问题,而且需要考虑子串的最小长度。这提示我们可能需要使用一种方式来逐步探索不同的子串。滑动窗口的适用性:滑动窗口是处理子串问题的常用技巧,特别是当…

【图像分割】nnUnetV1与V2的Linux部署与应用命令

以前觉得麻烦,一直没用过nnunet,虽然知道它很火,最近一个契机,部署使用了一下nnunet,记录一下其部署和使用的方法与命令。 1、部署 首先,我有一个环境,这个环境可以是以前就有的,也可…

1.c++入门(命名空间、缺省参数、函数重载、引用、内联函数、for循环、auto关键字、指针空值nullptr)

1.c的第一个程序 // 方法一 #include<iostream>// namespace为命名空间的关键字&#xff0c;std为空间名&#xff1b; C标准库的东西放进std命名空间 using namespace std; int main() {cout << "hello world" << endl;return 0; }// 方法二 #in…

YOLOv8的多分类模型如何计算准确率(Accuracy)、精确率(Precision)、召回率(recall)和F1-Score模型评估参数

《博主简介》 小伙伴们好&#xff0c;我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。 ✌更多学习资源&#xff0c;可关注公-仲-hao:【阿旭算法与机器学习】&#xff0c;共同学习交流~ &#x1f44d;感谢小伙伴们点赞、关注&#xff01; 《------往期经典推…

【Linux】 OpenSSH_9.3p1 升级到 OpenSSH_9.6p1(亲测无问题,建议收藏)

&#x1f468;‍&#x1f393;博主简介 &#x1f3c5;CSDN博客专家   &#x1f3c5;云计算领域优质创作者   &#x1f3c5;华为云开发者社区专家博主   &#x1f3c5;阿里云开发者社区专家博主 &#x1f48a;交流社区&#xff1a;运维交流社区 欢迎大家的加入&#xff01…

【leetCode】2810. 故障键盘

文章目录 [2810. 故障键盘](https://leetcode.cn/problems/faulty-keyboard/)思路一&#xff1a;模拟代码&#xff1a;思路二&#xff1a;双端队列代码&#xff1a; 2810. 故障键盘 思路一&#xff1a;模拟 用StringBuilder来拼贴字符遍历字符串&#xff0c;如果遇到i,对拼贴好…

注解,自定义注解和元注解

1.注解 1.1.注解概述、作用 注解&#xff08;Annotation&#xff09;&#xff0c;也叫元数据。一种代码级别的说明。它是JDK1.5及以后版本引入的一个特性&#xff0c;与类、接口、枚举是在同一个层次。它可以声明在包、类、字段、方法、局部变量、方法参数等的前面&#xff0…