当前位置: 首页 > news >正文

【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

  
  在这里插入图片描述

相关文章:

【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…...

JVM字节码与类的加载——class文件结构

文章目录 1、概述1.1、class文件的跨平台性1.2、编译器分类1.3、透过字节码指令看代码细节 2、虚拟机的基石:class文件2.1、字节码指令2.2、解读字节码方式 3、class文件结构3.1、魔数:class文件的标识3.2、class文件版本号3.3、常量池:存放所…...

小程序如何通过公众号发送新订单提醒

当客户在小程序上下单后,公众号会发送订单通知,这可以让管理员及时获知用户下单情况,方便及时处理订单和提供服务。下面是具体介绍如何设置公众号来发送订单服务通知。 方式一:通过采云公众号发送订单通知 此种方式是默认的通知…...

聊聊公众号最让我不爽的两个痛点

本文首发于 Python猫 微信公众号最让我不爽的地方有两个,而且有很多人虽然也不爽,却不知道原因。 本文想聊聊公众号的两个痛点,因为我经常收到私信问这两个问题,本文算是一次集中的回复吧。 第一个不爽的点是公众号会屏蔽外链&…...

【leetCode】2810. 故障键盘

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

xshell7连接ubuntu18.04

🎡导航小助手🎡 1.查看ubuntu IP2.开启openssh-server3.静态IP设置4.Xshell连接 1.查看ubuntu IP 输入下面命令查看IP ifconfig -a可以看到网卡是ens33,IP为192.168.3.180。 2.开启openssh-server 1、执行下句,下载SSH服务 s…...

真正的力量:实力与人际关系的平衡艺术

在当今社会,人们常常在追求个人发展和建立良好人际关系之间寻找平衡。有一种观点认为,“没有实力,就不要对别人好。不然,很容易被定义为讨好。”这句话在一定程度上揭示了实力与人际关系之间的微妙联系。本文将探讨这一观点的深层…...

Acwing.1388 游戏(区间DP对抗思想)

题目 玩家一和玩家二共同玩一个小游戏。 给定一个包含 N个正整数的序列。 由玩家一开始,双方交替行动。 每次行动可以在数列的两端之中任选一个数字将其取走,并给自己增加相应数字的分数。(双初始分都是 0分) 当所有数字都被…...

Numpy数组转换为csv文件

参考:Converting Numpy Array to CSV 在数据分析和处理中,经常会涉及到将数据从一个形式转换为另一个形式的操作。 其中,将Numpy数组转换为csv文件是一种常见的操作,因为csv文件是一种通用的数据存储格式,方便与其他软…...

替代安全指标(Surrogate Safety Measures (SSM) )

替代安全措施(Surrogate Safety Measures (SSM) )用于从数据中寻找接近碰撞,或可能发生(但实际没有发生)的碰撞事件。 SSM的两个合格标准: (1)它应该来自与碰撞直接相关的交通冲突&…...

usb_camera传输视频流编码的问题记录!

前言: 大家好,今天给大家分享的内容是,一个vip课程付费的朋友,在学习过程中遇到了一个usb采集的视频数据流,经过ffmpeg编码,出现了问题: 问题分析: 其实这个问题不难,关键…...

Linux安装nginx保姆级教程

文章目录 前言一、nginx安装(保姆级教程)1.安装nginx依赖2.安装wget3.创建nginx安装目录4.下载nginx5.查看下载好的nginx6.解压缩7.查看当前目录下的文件→进入nginx-1.8.0目录→查看当前目录下的文件8.安装nginx9.查看nginx安装目录并启动nginx10.网络请…...

leetcode-判断二分图

. - 力扣(LeetCode) 存在一个 无向图 ,图中有 n 个节点。其中每个节点都有一个介于 0 到 n - 1 之间的唯一编号。给你一个二维数组 graph ,其中 graph[u] 是一个节点数组,由节点 u 的邻接节点组成。形式上&#xff0c…...

算法day30 回溯6

332 重新安排行程 给你一份航线列表 tickets ,其中 tickets[i] [fromi, toi] 表示飞机出发和降落的机场地点。请你对该行程进行重新规划排序。 所有这些机票都属于一个从 JFK(肯尼迪国际机场)出发的先生,所以该行程必须从 JFK …...

分享three.js实现乐高小汽车

前言 Web脚本语言JavaScript入门容易,但是想要熟练掌握却需要几年的学习与实践,还要在弱类型开发语言中习惯于使用模块来构建你的代码,就像小时候玩的乐高积木一样。 应用程序的模块化理念,通过将实现隐藏在一个简单的接口后面&a…...

gpt的构造和原理

gpt是序列预测模型。 问答是通过确定问答格式样本训练出来的!比如“Q:xxxx.A:xxx"本质还是根据前面的序列预测后面的序列。在自回归训练过程中,文本序列(可能包含问题和紧随其后的答案)被视为一个整体输入到模型…...

基于springboot实现教师人事档案管理系统项目【项目源码+论文说明】计算机毕业设计

基于springboot实现IT技术交流和分享平台系统演示 摘要 我国科学技术的不断发展,计算机的应用日渐成熟,其强大的功能给人们留下深刻的印象,它已经应用到了人类社会的各个层次的领域,发挥着重要的不可替换的作用。信息管理作为计算…...

K8S之Job和CronJob控制器

这里写目录标题 Job概念适用场景使用案例 CronJob概念适用场景使用案例 Job 概念 Job控制器用于管理Pod对象运行一次性任务,例如:对数据库备份,可以直接在k8s上启动一个mysqldump备份程序,也可以启动一个pod,这个pod…...

基于SSM的基于个人需求和地域特色的外卖推荐系统(有报告)。Javaee项目。ssm项目。

演示视频: 基于SSM的基于个人需求和地域特色的外卖推荐系统(有报告)。Javaee项目。ssm项目。 项目介绍: 采用M(model)V(view)C(controller)三层体系结构&…...

哈佛大学商业评论 --- 第三篇:真实世界中的增强现实

AR将全面融入公司发展战略! AR将成为人类和机器之间的新接口! AR将成为人类的关键技术之一! 请将此文转发给您的老板! --- 本文作者:Michael E.Porter和James E.Heppelmann 虽然物理世界是三维的,但大…...

脑机新手指南(八):OpenBCI_GUI:从环境搭建到数据可视化(下)

一、数据处理与分析实战 (一)实时滤波与参数调整 基础滤波操作 60Hz 工频滤波:勾选界面右侧 “60Hz” 复选框,可有效抑制电网干扰(适用于北美地区,欧洲用户可调整为 50Hz)。 平滑处理&…...

大语言模型如何处理长文本?常用文本分割技术详解

为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...

Psychopy音频的使用

Psychopy音频的使用 本文主要解决以下问题: 指定音频引擎与设备;播放音频文件 本文所使用的环境: Python3.10 numpy2.2.6 psychopy2025.1.1 psychtoolbox3.0.19.14 一、音频配置 Psychopy文档链接为Sound - for audio playback — Psy…...

【C++进阶篇】智能指针

C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...

三分算法与DeepSeek辅助证明是单峰函数

前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...

深度学习之模型压缩三驾马车:模型剪枝、模型量化、知识蒸馏

一、引言 在深度学习中,我们训练出的神经网络往往非常庞大(比如像 ResNet、YOLOv8、Vision Transformer),虽然精度很高,但“太重”了,运行起来很慢,占用内存大,不适合部署到手机、摄…...

【1】跨越技术栈鸿沟:字节跳动开源TRAE AI编程IDE的实战体验

2024年初,人工智能编程工具领域发生了一次静默的变革。当字节跳动宣布退出其TRAE项目(一款融合大型语言模型能力的云端AI编程IDE)时,技术社区曾短暂叹息。然而这一退场并非终点——通过开源社区的接力,TRAE在WayToAGI等…...

未授权访问事件频发,我们应当如何应对?

在当下,数据已成为企业和组织的核心资产,是推动业务发展、决策制定以及创新的关键驱动力。然而,未授权访问这一隐匿的安全威胁,正如同高悬的达摩克利斯之剑,时刻威胁着数据的安全,一旦触发,便可…...

【Redis】Redis从入门到实战:全面指南

Redis从入门到实战:全面指南 一、Redis简介 Redis(Remote Dictionary Server)是一个开源的、基于内存的键值存储系统,它可以用作数据库、缓存和消息代理。由Salvatore Sanfilippo于2009年开发,因其高性能、丰富的数据结构和广泛的语言支持而广受欢迎。 Redis核心特点:…...

XXE漏洞知识

目录 1.XXE简介与危害 XML概念 XML与HTML的区别 1.pom.xml 主要作用 2.web.xml 3.mybatis 2.XXE概念与危害 案例:文件读取(需要Apache >5.4版本) 案例:内网探测(鸡肋) 案例:执行命…...