指针和数组笔试题讲解(3)

news2025/1/23 7:15:58

🐵本篇文章将对指针相关笔试题进行讲解,同时也是指针和数组笔试题讲解的最后一篇文章,那么接下来将会对8道笔试题进行逐一讲解

笔试题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;
}

1.1代码分析📜

此代码先定义了一个长度为5的整形数组a,接下来取出整个数组的地址再+1,之后强转为int*类型并传给指针ptr,此时整形指针指向的位置如下:

题目分别要求*(a + 1)和*(ptr - 1),a为数组首元素的地址,+1后为第二个元素的地址,再解引用后就是2,ptr指向的地址-1后,指向5的地址,再解引用后就是5

1.2打印结果🖨️

笔试题2💻

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

    return 0;
}

2.1代码分析📜

此代码定义了一个结构体指针,且题目说明结构体Test类型的变量大小是20个字节;

p + 0x1:p是一个结构体指针,指针在进行+-操作时会跳过其所对应类型的字节数,比如p如果是一个整形指针,那么+1后就跳过4个字节,在该题,p是一个结构体指针,+1就跳过20个字节,题目中加的是0x1,十六进制的1和十进制的1一样,所以p跳过20个字节,转换为16进制后为0x100014;

(unsigned long)p + 0x1):将p强转为了无符号长整形类型,那么此时p就不再是一个指针而是一个无符号长整形变量,+1后就只是+1个字节,之后再以%p(以十六进制的形式打印地址)打印p,结果为0x100001

(unsigned int*)p + 0x1:将p强转为了无符号整形指针类型,此时p是一个整形指针类型的变量,+1后跳过4个字节,结果为0x100004

2.2打印结果🖨️

以下是x86环境下运行的结果

笔试题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;
}

3.1代码分析📜

此代码定义了一个长度为4的整形数组a,取出整个数组的地址后+1,再强转为int*类型赋给ptr1,此时ptr1指向如下位置:

printf函数中prt[-1]相当于*(ptr1 - 1),以十六进制的形式打印的话,答案就是4

int* ptr2 = (int*)((int)a + 1);涉及到了数据在内存中存储的相关内容,在之前的深度剖析数据在内存中的存储这篇文章中,讲到过大小端存储模式;

大端存储模式:是指数据的低位保存在内存的高地址中,而数据的高位,保存在内存的低地址中

小端存储模式:是指数据的低位保存在内存的低地址中,而数据的高位,保存在内存的高地址中

具体是大端还是小端取决于编译器,在vs2022中是小端存储模式

在vs2022中数组a在内存中的存储方式如下(数据都是以16进制形式表示的):

a是数组首元素的地址,类型为int*,+1就会跳过4个字节,但是强转为int后+1就变成了一个整形变量,+1就只跳过1个字节

由于是小端存储模式,所以ptr2解引用后得到的十六进制数据就是20000000

3.2打印结果🖨️

笔试题4💻

int main()
{
	int a[3][2] = { (0, 1), (2, 3), (4, 5) }; //注意这里应该是大括号而不是小括号
	int* p;
	p = a[0];
	printf("%d", p[0]);
	return 0;
}

4.1代码分析📜

首先你应该注意到了一个小陷阱,那就是二维数组a中用了小括号而不是大括号,那么每个小括号内部就是一个逗号表达式,其结果分别为1,3,5;之后将第一行一维数组的地址传给了p:

之后求p[0]相当于*(p + 0),也就是1

4.2打印结果🖨️

笔试题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;
}

5.1代码分析📜

a是一个5行5列的二维数组,p是一个指向具有4个整型元素的一维数组的指针,之后将a的第一行的地址赋给了p

&p[4][2]和&a[4][2]都是int*类型的指针,指针-指针可以实现两个指针间元素的个数,那么以%d的形式打印&p[4][2] - &a[4][2]的结果为-4;%p用来以十六进制的形式打印地址,实际上就是打印指针的值,但如果打印的不是指针,就会打印其十六进制形式, -4不是指针,所以答案是其十六进制:FFFFFFFC

5.2打印结果🖨️

笔试题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));//10 5
	return 0;
}

6.1代码分析📜

先看int* ptr1 = (int*)(&aa + 1);将整个数组的地址取出再+1后指针跳过整个数组,再强转为int*传给指针ptr1

ptr1-1后向前走四个字节,指向10的地址,解引用后就是10,所以*(ptr1 - 1) = 10

再看int* ptr2 = (int*)(*(aa + 1));*(aa + 1)等价于aa[1],也就是第二行的数组名,也就是第二行数组首元素的地址,强转为int*后传给指针ptr2

ptr2-1后向前走四个字节,指向5的地址,解引用后就是5,所以*(ptr2 - 1) = 5

6.2打印结果🖨️

笔试题7💻

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

7.1代码分析📜

a是一个字符指针数组,而每一个字符指针指向其所对应的字符串的首地址,之后将数组首元素地址也就是字符指针的地址传给了二级指针pa,++后指向第二个字符指针的地址,在printf函数中pa进行了一次解引用,那*pa就是第二个字符指针,第二个字符指针指向'a'的地址,那么*pa就是'a'的地址,以%s打印后为at

7.2打印结果🖨️

笔试题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;
}

8.1代码分析📜

三个变量的定义如上图所示

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

cpp在++后指向c+2的地址,第一次解引用后变为c+2,c+2指向c的第三个元素的地址,cpp再解引用后就是c的第三个字符指针,此指针指向'P'的地址,按%s打印后就是POINT

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

cpp在++后指向c+1(前面++过了一次)第一次解引用后变为c+1,c+1指向c的第二个元素的地址,--后指向c的第一个元素的地址,cpp再解引用后就是c的第一个字符指针,此指针指向'E'的地址,+3后指向'E'的地址,按%s打印后就是ER

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

cpp-2指向c+3的地址,第一次解引用后变为c+3,c+3指向c的第四个元素的地址,cpp在解引用后就是c的第四个字符指针,此指针指向'F'的地址,+3后指向'S'的地址,按%s打印后就是ST

printf("%s\n", cpp[-1][-1] + 1); //相当于*(*(cpp - 1) - 1) + 1

cpp-1指向c+2的地址,解引用后变为c+2,c+2指向c的第三个元素地址,减1再解引用后变为c的第二个字符指针,此指针指向'N'的地址,+1后指向'E'的地址,按%s打印后就是EW

8.2打印结果🖨️


🙉至此,指针的8道笔试题全部讲解完毕,指针和数组笔试题讲解系列也告一段落

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

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

相关文章

C#调用C++ dll 返回数组

先看一下C语言函数返回数组的问题&#xff1b; 先看一个错误的示范&#xff1b; 因为 a 是局部变量&#xff0c;只存在函数 function() 中&#xff0c;返回给main中的b是错误的&#xff1b; 函数返回数组的一种写法如下&#xff1b; #include<stdio.h> int function(in…

二维凸包(Graham) 模板 + 详解

&#xff08;闲话&#xff09; 上了大学后没怎么搞oi&#xff0c;从土木跑路到通信了&#xff08;提桶开润大成功&#xff01;&#xff09;&#xff0c;但是一年上两年的课&#xff08;补的&#xff09;&#xff0c;保研也寄掉了&#xff08; 说起来自从博客被大学同学发现并…

地牢大师问题(bfs提高训练 + 免去边界处理的特殊方法)

地牢大师问题 文章目录 地牢大师问题前言题目描述题目分析输入处理移动方式【和二维的对比】边界判断问题的解决 代码总结 前言 在之前的博客里面&#xff0c;我们介绍了bfs 基础算法的模版和应用,这里我们再挑战一下自己&#xff0c;尝试一个更高水平的题目&#xff0c;加深一…

vue2——电商项目 黑马

创建项目 初始化 router app.vue vant 组件库 Viewport 布局 vw适配 路由配置 底部导航组件 二级路由配置 登录页面 新建默认样式 main.js 引入commonless 登录静态页面—头部组件NavBar 导入navbar 引用 axios封装 图形验证码 获取 get 渲染 api接口模块 toast轻提示 使用 …

Flutter图标

https://fluttericon.cn/ Flutter 内置了丰富的图标。 Icon(Icons.ac_unit)

智能批量重命名,轻松删除文件名后缀数字并添加编号!

亲爱的用户们&#xff0c;您是否曾经为繁琐而重复的文件重命名工作而感到头疼&#xff1f;现在&#xff0c;我们为您提供一款智能化的工具&#xff0c;让文件重命名变得如此简单&#xff01; 首先&#xff0c;我们要进入文件批量改名高手&#xff0c;并在板块栏里选择“文件批…

overleaf 插入图片,引用图片,图标标题Fig与文章引用Figure不一致解决

目录 1.一般插图 2.插入双栏图片 3 插入子图 4. 引用出现问题 问题1 &#xff1a; pdf 文中引用只出现了图片序号&#xff0c;如“3”。没有出现“Fig.3 或者Figure.3” 问题2&#xff1a;文中引用的标题和图片下面的标题不一致 1 首先&#xff0c;在导言区添加以下行…

七天学会C语言-第二天(数据结构)

1. If 语句&#xff1a; If 语句是一种条件语句&#xff0c;用于根据条件的真假执行不同的代码块。它的基本形式如下&#xff1a; if (条件) {// 条件为真时执行的代码 } else {// 条件为假时执行的代码 }写一个基础的If语句 #include<stdio.h> int main(){int x 10;…

HarmonyOS开发环境搭建

一 鸿蒙简介&#xff1a; 1.1 HarmonyOS是华为自研的一款分布式操作系统&#xff0c;兼容Android&#xff0c;但又区别Android&#xff0c;不仅仅定位与手机系统。更侧重于万物物联和智能终端&#xff0c;目前已更新到4.0版本。 1.2 HarmonyOS软件编程语言是ArkTS&#xff0c…

STM32DMA原理和应用

目录 1.什么是DMA 2.DMA的意义 3.DMA搬运的数据和方式 4.DMA 控制器和通道 5.DMA通道的优先级 6.DMA传输方式 7.DMA应用 实验一: 内存到内存搬运 CubeMX配置&#xff1a; ​编辑用到的库函数&#xff1a; 代码实现思路&#xff1a; 实验二: 内存到外设搬运 CubeMX…

简单返回封装实体类(RespBean)

RespBean的作用 返回状态码&#xff0c;返回信息&#xff0c;返回数据 package com.example.entity;import lombok.AllArgsConstructor; import lombok.Data; import lombok.NoArgsConstructor;Data AllArgsConstructor NoArgsConstructor public class RespBean {private lon…

基于springboot实现的极验校验

概述 在系统业务中&#xff0c;需要想客户发送手机验证码&#xff0c;进行验证后&#xff0c;才能提交。但为了防止不正当的短信发送&#xff08;攻击&#xff0c;恶意操作等&#xff09;&#xff0c;需要在发送短信前添加一个行为验证&#xff08;这里使用的是极验&#xff0…

利用Python将dataframe格式的所有列的数据类型转换为分类数据类型

一、样例理解 import pandas as pd import numpy as np# 创建测试数据 feature_names [col1 , col2, col3, col4, col5, col6] values np.random.randint(20, size(10,6))dataset pd.DataFrame(data values, columns feature_names)print("转换前的数据为\n",d…

【C进阶】指针和数组笔试题解析

做题之前我们先来回顾一下 对于数组名的理解&#xff1a;除了以下两种情况&#xff0c;数组名表示的都是数组首元素的地址 &#xff08;1&#xff09;sizeof&#xff08;数组名&#xff09;&#xff1a;这里的数组名表示整个数组 &#xff08;2&#xff09;&&#xff08;数…

Maven3.6.1下载和详细配置

1.下载maven 说明&#xff1a;以下载maven3.6.1为例 1.1网址 Maven – Welcome to Apache Maven 1.2点击下载 1.3点击Maven 3 archives 1.4 点击相应的版本 1.5 点击binaries下载 说明&#xff1a;binaries是二进制的意思 1.6点击zip格式 1.7 蓝奏云获取 说明&#xff1a…

C语言——自定义类型结构体_学习笔记

结构体的基本概念 结构体是一种用户自定义的数据类型&#xff0c;可以包含多个不同类型的变量。通过使用结构体&#xff0c;我们可以将相关联的数据组织在一起&#xff0c;便于管理和使用。 结构体的声明 正常的结构体声明 在C语言中&#xff0c;结构体(struct)指的是一种数…

【SA8295P 源码分析】97 - QNX AIS Camera 框架介绍 及 Camera 工作流程分析

【SA8295P 源码分析】97 - QNX AIS Camera 框架介绍 及 Camera 工作流程分析 一、QNX AIS Server 框架分析二、QNX Hypervisor / Android GVM 方案介绍三、Camera APP 调用流程分析四、QCarCam 状态转换过程介绍五、Camera 加串-解串 硬件链路分析六、摄像头初始化检测过程介绍…

Debian环境下搭建STM32开发环境

1. 安装交叉编译工具&#xff0c;解压gcc-arm-none-eabi-10.3-2021.10-x86_64-linux.tar.bz2&#xff0c;并且把交叉编译环境添加到path路径。 2.安装下载工具驱动和下载工具 # 安装下载工具openocd sudo apt -y install openocd 3.下载测试 sudo openocd -f cmsis-dap.cfg -…

吴恩达ChatGPT《Finetuning Large Language Models》笔记

课程地址&#xff1a;https://learn.deeplearning.ai/finetuning-large-language-models/lesson/1/introduction Introduction 动机&#xff1a;虽然编写提示词&#xff08;Prompt&#xff09;可以让LLM按照指示执行任务&#xff0c;比如提取文本中的关键词&#xff0c;或者对…

postgresql|数据库|centos7下基于postgresql-12的主从复制的pgpool-4.4的部署和使用

前言&#xff1a; postgresql数据库只用自身的一些配置是无法做到最优的优化的&#xff0c;需要通过一些外置插件&#xff08;中间件&#xff09;来提高服务器的整体性能&#xff0c;通俗的说就是数据库仅仅依靠自身是无法达到性能最优的&#xff0c;很多时候需要更改数据库的…