32 C 语言指针的长度与运算(指针加减整数、指针自增自减、同类型指针相减、指针的比较运算)
目录
1 指针的长度
2 指针与整数的加减运算
3 指针自增与自减
4 同类型指针相减
5 指针的比较运算
6 测试题
1 指针的长度
在 C 语言中,sizeof 运算符可以用来计算指针的长度。指针的长度实际上与指针所指向的数据类型无关,而是与系统的位数(即系统架构)有关。具体来说:
- 32 位系统:无论指针指向什么类型的数据(如 int、char、float 等),指针的长度都是 4 个字节。
- 64 位系统:无论指针指向什么类型的数据,指针的长度都是 8 个字节。
下面是一个示例,展示了如何使用 sizeof 运算符来计算不同类型指针的长度:
#include <stdio.h>int main()
{int *int_ptr;char *char_ptr;float *float_ptr;double *double_ptr;printf("int 指针的长度: %zu 字节\n", sizeof(int_ptr));printf("char 指针的长度: %zu 字节\n", sizeof(char_ptr));printf("float 指针的长度: %zu 字节\n", sizeof(float_ptr));printf("double 指针的长度: %zu 字节\n", sizeof(double_ptr));return 0;
}
在 32 位系统上运行上述代码,输出可能是:
int 指针的长度: 4 字节
char 指针的长度: 4 字节
float 指针的长度: 4 字节
double 指针的长度: 4 字节
在 64 位系统上运行上述代码,输出可能是:
int 指针的长度: 8 字节
char 指针的长度: 8 字节
float 指针的长度: 8 字节
double 指针的长度: 8 字节
结论:
- 指针的长度与系统架构有关:在 32 位系统中,所有指针的长度都是 4 个字节;在 64 位系统中,所有指针的长度都是 8 个字节。
- 指针的长度与指向的数据类型无关:无论指针指向 int、char、float 还是 double,指针本身的长度是固定的,由系统的位数决定。
2 指针与整数的加减运算
指针与整数的加减运算表示指针所指向的内存地址的移动。具体来说:
- 加法运算:指针加一个整数表示指针所指向的内存地址向后移动。
- 减法运算:指针减一个整数表示指针所指向的内存地址向前移动。
指针移动的步长与指针指向的数据类型有关。每移动一个单位,指针会移动相应数据类型所占的字节数。例如:
- 如果指针指向 int 类型的数据,int 通常占用 4 个字节,那么指针加 1 会向后移动 4 个字节,指针减 2 会向前移动 8 个字节。
- 如果指针指向 char 类型的数据,char 通常占用 1 个字节,那么指针加 1 会向后移动 1 个字节,指针减 2 会向前移动 2 个字节。
数组的元素在内存中是连续存储的,因此通过数组元素可以很好地演示指针加减整数的情况。以下是一个示例:
#include <stdio.h>int main()
{// 创建一个包含 5 个整数的数组int nums[] = {10, 20, 30, 40, 50};// 创建一个指针并将其初始化为数组第一个元素的地址int *ptr = &nums[0];// int *ptr = nums; 或者直接指向数组名 和上面等价// 打印指针的地址和指针所指向的值printf("初始状态: ptr=%p, *ptr=%d \n", (void *)ptr, *ptr);// 指针加 3,指针指向 int 类型,每个 int 占 4 个字节// 因此,指针会向后移动 3 * 4 = 12 个字节ptr += 3;printf("指针加 3 后: ptr=%p, *ptr=%d \n", (void *)ptr, *ptr);// 指针减 2,指针会向前移动 2 * 4 = 8 个字节ptr -= 2;printf("指针减 2 后: ptr=%p, *ptr=%d \n", (void *)ptr, *ptr);return 0;
}
输出结果如下所示:
3 指针自增与自减
指针的自增和自减本质上是通过加减整数来实现的。自增会使指针向后移动,自减会使指针向前移动。移动的步长与指针指向的数据类型有关。每移动一个单位,指针会移动相应数据类型所占的字节数。例如,如果指针指向 short 类型的数据,short 通常占用 2 个字节,那么指针自增 1 会向后移动 2 个字节,指针自减 1 会向前移动 2 个字节。
#include <stdio.h>int main()
{// 创建数组,元素都是 short 类型,每个元素占据 2 个字节short nums[] = {10, 20, 30, 40, 50};// 定义常量记录数组长度const int len = sizeof(nums) / sizeof(nums[0]); // 5// 利用指针自增遍历数组元素// 创建指针并指向数组第一个元素的地址short *ptr = &nums[0];// short *ptr = nums; 或者直接指向数组名 和上面等价// 循环遍历数组for (int i = 0; i < len; i++, ptr++){// 打印当前元素的索引、地址和值printf("元素索引:%d, 元素地址:%p, 元素值:%hd \n", i, (void *)ptr, *ptr);// 指针自增,向后移动一个 short 类型的单位(2个字节)// ptr++; 可以放到循环变量迭代里面去}printf("\n");// 循环遍历数组,从最后一个元素到第一个元素// 此时指针超出数组界限,需先自减一次for (int i = len - 1; i >= 0; i--){// 指针自减,向前移动一个 short 类型的单位(2个字节)ptr--;// 打印当前元素的索引、地址和值printf("元素索引:%d, 元素地址:%p, 元素值:%hd \n", i, (void *)ptr, *ptr);}return 0;
}
输出结果如下所示:
在上述示例中,当第一次循环(正序输出数组元素)结束后,指针 ptr 已经超出了数组的边界,指向了数组最后一个元素之后的位置,如下图所示。因此,在进行第二次循环(倒序输出数组元素)之前,需要先将指针重置为数组最后一个元素的地址,以确保能够正常输出。
可以使用 ptr-- 重置指针:在第一次循环结束后,指针 ptr 超出了数组的边界。可以通过 ptr-- 将指针向前移动一个单位,使其指向数组的最后一个元素。如上述代码中的第二次循环一开始就使用 ptr-- 重置指针。
也可以直接修改指针的指向地址:可以直接将指针 ptr 重新初始化为数组最后一个元素的地址,这样可以确保指针指向正确的起始位置。如下代码所示:
#include <stdio.h>int main()
{// 创建一个包含 5 个 short 类型的数组short nums[] = {10, 20, 30, 40, 50};int n = sizeof(nums) / sizeof(nums[0]); // 计算数组的长度// 创建指针并初始化为数组第一个元素的地址short *ptr = &nums[0];// 正序输出数组元素printf("正序输出数组元素:\n");for (int i = 0; i < n; i++){printf("元素索引:%d, 元素地址:%p, 元素值:%hd \n", i, (void *)ptr, *ptr);ptr++; // 指针自增,向后移动一个 short 类型的单位(2个字节),可以放到循环变量迭代里面去}printf("\n");// 重置指针,使其指向数组最后一个元素的地址ptr = &nums[n - 1];// 倒序输出数组元素printf("倒序输出数组元素:\n");for (int i = n - 1; i >= 0; i--){printf("元素索引:%d, 元素地址:%p, 元素值:%hd \n", i, (void *)ptr, *ptr);ptr--; // 指针自减,向前移动一个 short 类型的单位(2个字节)}return 0;
}
4 同类型指针相减
相同类型的指针可以进行减法运算,返回它们之间的距离,即相隔多少个数据单位。高位地址减去低位地址,返回的是正值;低位地址减去高位地址,返回的是负值。同类型指针相减的结果是一个 ptrdiff_t 类型的数据,ptrdiff_t 是一个带符号的整数,格式输出中对应的格式占位符是 %td。
以下是一个示例,演示了如何使用同类型指针相减来计算它们之间的距离:
#include <stdio.h>int main()
{// 创建一个包含 5 个整数的数组int nums[] = {10, 20, 30, 40, 50};// 创建指针并指向数组第一个元素的地址int *ptr1 = &nums[0];// 创建指针并指向数组第四个元素的地址int *ptr2 = &nums[3];// 打印查看原始内容printf("ptr1地址:%p\n", ptr1);printf("ptr2地址:%p\n", ptr2);// 计算两个指针之间的距离// ptr2 - ptr1 应该等于 3,因为 ptr2 指向第四个元素,ptr1 指向第一个元素printf("ptr2 - ptr1 = %td \n", ptr2 - ptr1); // 输出 3// ptr1 - ptr2 应该等于 -3,因为 ptr1 指向第一个元素,ptr2 指向第四个元素printf("ptr1 - ptr2 = %td \n", ptr1 - ptr2); // 输出 -3// 再连续创建两个变量double d1 = 1.0;double d2 = 2.0;// 创建指针并分别指向 d1 和 d2 的地址double *p1 = &d1;double *p2 = &d2;printf("p1地址:%p\n", p1);printf("p2地址:%p\n", p2);// 计算两个指针之间的距离// p1 - p2 应该等于 1,因为 p1 指向 d1,p2 指向 d2,d1 和 d2 在内存中是相邻的printf("p1 - p2 = %td \n", p1 - p2); // 输出 1// p2 - p1 应该等于 -1,因为 p2 指向 d2,p1 指向 d1,d1 和 d2 在内存中是相邻的printf("p2 - p1 = %td \n", p2 - p1); // 输出 -1return 0;
}
输出结果如下所示:
5 指针的比较运算
指针之间可以进行比较运算,如 ==、!=、<、<=、>、>=。这些运算符比较的是指针所指向的内存地址的大小,返回值是 int 类型的整数,1 表示 true,0 表示 false。
以下是一个示例,演示了如何使用指针的比较运算:
#include <stdio.h>int main()
{// 创建一个包含 5 个整数的数组int nums[] = {10, 20, 30, 40, 50};double n = 1.0;// 创建指针并指向数组第一个元素的地址int *ptr1 = &nums[0];// 创建指针并指向数组第四个元素的地址int *ptr2 = &nums[3];// 创建指针也指向数组第一个元素的地址int *ptr3 = &nums[0];// 创建指针指向变量 n 的地址double *ptr4 = &n;// 输出指针指向的地址printf("ptr1=%p\n", (void *)ptr1);printf("ptr2=%p\n", (void *)ptr2);printf("ptr3=%p\n", (void *)ptr3);printf("ptr4=%p\n\n", (void *)ptr4);// 进行比较// 比较 ptr1 和 ptr2 的地址printf("ptr1 > ptr2: %d \n", ptr1 > ptr2); // 比较 ptr1 是否大于 ptr2printf("ptr1 < ptr2: %d \n", ptr1 < ptr2); // 比较 ptr1 是否小于 ptr2// 比较 ptr1 和 ptr3 的地址printf("ptr1 == ptr3: %d \n", ptr1 == ptr3); // 比较 ptr1 是否等于 ptr3// 比较不同类型的指针(ptr4 和 ptr1)// 注意:不同类型的指针进行比较会引发编译器警告printf("ptr4 > ptr1: %d \n", ptr4 > ptr1); // 比较 ptr4 是否大于 ptr1return 0;
}
注意,不同类型的指针进行比较会引发编译器警告。如下所示:
输出结果如下所示:
6 测试题
1. 请写出下面程序的运行结果(64 位操作系统)。
int num = 250;
int *p = #
printf("%zu \n", sizeof p);
printf("%zu \n", sizeof *p);
【答案】
8
4
【解析】
sizeof p 是计算指针本身的长度,指针存储的是地址,在 64 位操作系统下,地址占 8 个字节。
sizeof *p 是计算指针指向的数据的长度,指针 p 指向 int 类型变量 num,int 类型长度是 4 个字节。
2. 请写出下面程序的运行结果。
int arr[] = {10,20,30,40,50};
int *p = arr;
printf("%d", *(p+1) + *(p+2));
【答案】
50
【解析】
指针 p 指向数组 arr 的首元素地址。
p+1 得到数组 arr 第二个元素的地址,*(p+1) 得到第二个元素的值 20。
p+2 得到数组 arr 第三个元素的地址,*(p+2) 得到第三个元素的值 30。
相关文章:

32 C 语言指针的长度与运算(指针加减整数、指针自增自减、同类型指针相减、指针的比较运算)
目录 1 指针的长度 2 指针与整数的加减运算 3 指针自增与自减 4 同类型指针相减 5 指针的比较运算 6 测试题 1 指针的长度 在 C 语言中,sizeof 运算符可以用来计算指针的长度。指针的长度实际上与指针所指向的数据类型无关,而是与系统的位数&…...

【系统架构设计师】经典论文:轮软件三层架构设计
更多内容请见: 备考系统架构设计师-核心总结目录 文章目录 摘要正文总结摘要 本人于 2022 年 1 月参与了中石化 XX 油田 XX 采油厂“用电管理系统”的项目建设,该系统建设目标是实现分单位、分线路、分系统评价、优化、考核,全面提升采油厂用 电管理水平。在该项目组中我担…...

(C语言贪吃蛇)13.实现贪吃蛇四方向的移动
目录 前言 原代码预览 解决方法⚠️ 运行效果 总结 前言 我们上节通过Linux线程实现了两个while(1)同时运行,这样就可以一边控制方向一遍刷新出贪吃蛇的身体节点了。本节我们就来实现贪吃蛇四方向的移动。 (此图片为最终效果) 原代码预览 我们之前的代码是通过…...

Spring Boot + MyBatis 项目中常用注解详解(万字长篇解读)
Spring Boot MyBatis 项目中常用注解详解 在现代Java开发中,Spring Boot和MyBatis是两大热门框架,广泛应用于构建高效、可维护的企业级应用。两者结合使用,可以充分发挥各自的优势,提高开发效率和代码质量。在这个过程中&#x…...

AWS Network Firewall -NAT网关配置只应许白名单域名出入站
1. 创建防火墙 选择防火墙的归属子网(选择公有子网) 2. 创建规则白名单域名放行 3. 绑定相关规则 继续往下拉 绑定非托管规则 4. 配置网络路由 相关规则 参考图 解释 防火墙的归属公有子网路由表规则机器实例的规则子网路由表规则nat网管路…...

【C语言系统编程】【第二部分:并发编程】2.3 实践与案例分析
2.3 实践与案例分析 2.3.1 案例分析:多线程文件搜索程序 本文中,我们将通过一个多线程文件搜索程序的案例,展示如何在实际项目中应用多线程编程技术,并具体介绍任务分解、线程创建、结果汇总及锁机制的应用。 2.3.1.1 任务分解…...

React -AppVarContext.Provider 提供者组件
AppVarContext.Provider 是一个 React 上下文提供者,通常用于在组件树中提供共享的状态或数据。下面将详细解释 AppVarContext.Provider 的作用和如何使用它。展示如何使用 AppVarContext.Provider 来管理全局状态 1. 什么是上下文(Context)…...

【Python】解密用户代理:使用 Python User Agents 库探索浏览器和设备信息
Python User Agents 是一个专为解析 User Agent 字符串而设计的 Python 库。它能够轻松识别访问设备的类型(如移动设备、桌面设备或平板),并获取设备、浏览器、操作系统等详细信息。借助它,开发者可以更好地了解访问用户的设备属性…...

以串口接口为例介绍关于BSP底层架构开发的迭代过程
以串口接口为例介绍关于BSP底层架构开发的迭代过程 文章目录 以串口接口为例介绍关于BSP底层架构开发的迭代过程架构概述初代BSP二代BSP:三代BSP:四代BSP:架构概述 单片机开发有四个阶段: 阶段一:单一单片机的功能实现阶段 此阶段你开始熟悉STM32F1系列的单片机,并利用…...

Label-Studio ML利用yolov8模型实现自动标注
引言 Label Studio ML 后端是一个 SDK,用于包装您的机器学习代码并将其转换为 Web 服务器。Web 服务器可以连接到正在运行的 Label Studio 实例,以自动执行标记任务。我们提供了一个示例模型库,您可以在自己的工作流程中使用这些模型&#x…...

【PostgreSQL】实战篇——用户管理、角色和权限控制的高级用法及技巧
数据库中用户管理、角色和权限控制不仅仅是基础的安全措施,更是实现复杂应用需求和优化数据库性能的重要手段。 通过深入理解这些概念,数据库管理员可以更有效地管理用户访问、确保数据安全,并优化系统性能。以下是对这些概念的详细介绍以及…...

Leetcode: 0011-0020题速览
Leetcode: 0011-0020题速览 本文材料来自于LeetCode solutions in any programming language | 多种编程语言实现 LeetCode、《剑指 Offer(第 2 版)》、《程序员面试金典(第 6 版)》题解 遵从开源协议为知识共享 版权归属-相同方式…...

Hive数仓操作(七)
一、 Hive动态分区表 1. 动态分区与静态分区的区别 分区定义: 静态分区:在插入数据时,需要手动指定分区字段的值。动态分区:分区字段的值是根据数据中的某个字段自动生成的,用户只需指定分区字段的类型。 数据加载方…...

Redis进阶篇 - 缓存穿透、缓存击穿、缓存雪崩问题及其解决方案
文章目录 1 文章概述2 缓存穿透2.1 什么是缓存穿透?2.2 缓存穿透的解决方法2.2.1 做好参数校验2.2.2 缓存无效Key2.2.3 使用布隆过滤器2.2.4 接口限流 3 缓存击穿3.1 什么是缓存击穿?3.2 缓存击穿的解决方法3.2.1 调整热点数据过期时间3.2.2 热点数据预热…...

一天认识一个硬件之电源
无论是台式机还是笔记本,都离不开电源,台式机和笔记本电脑的电源都承担着将交流电转换为直流电,并为电脑内部各个部件提供稳定电力供应的重要任务。今天就来分享一下台式机和笔记本的电源区别 设计和功率 台式机电源:设计为内置…...

关于BSV区块链覆盖网络的常见问题解答(上篇)
发表时间:2024年9月20日 在BSV区块链上的覆盖网络服务为寻求可扩展、安全、高效交易处理解决方案的开发者和企业家开辟了新的视野。 作为开创性的曼达拉升级的一部分,覆盖网络服务提供了一个强大的框架,用于管理特定类型的交易和数据访问…...

VUE 开发——Node.js学习(一)
一、认识Node.js Node.js是一个跨平台JavaScript运行环境,使开发者可以搭建服务器端的JavaScript应用程序 使用Node.js编写服务器端程序——编写数据接口、前端工程化; Node.js环境没有BOM和DOM; Node.js安装:下载node-v16.19…...

角膜移植难题现,传统方式缺陷显,创新水凝胶破局
大家好!今天来了解一篇天然聚合物衍生光固化生物粘附水凝胶研究——《Natural polymer-derived photocurable bioadhesive hydrogels for sutureless keratoplasty》发表于《Bioactive Materials》。本文介绍了一种用于无缝合角膜移植术的天然聚合物衍生光固化生物粘…...

探索Spring Boot:实现“衣依”服装电商平台
1系统概述 1.1 研究背景 如今互联网高速发展,网络遍布全球,通过互联网发布的消息能快而方便的传播到世界每个角落,并且互联网上能传播的信息也很广,比如文字、图片、声音、视频等。从而,这种种好处使得互联网成了信息传…...

使用 cron 来设置定时任务
使用 cron 来设置定时任务,以便定期运行这个脚本。 在 Linux 系统中设置定时任务的步骤: 1. 编辑 cron 任务 打开终端并输入以下命令以编辑 crontab 文件: crontab -e2. 设置每天 8 点运行脚本 在 crontab 文件的最后,添加以下…...

C# Blazor Server 调用海康H5Player播放摄像头画面
目标 调用海康综合安防平台api,通过摄像头的cameraIndexCode调用【获取监控点预览取流URLv2】api,得到websocket 的url,然后在blazor server中使用htplayer.js播放摄像头实时画面。 步骤 根据摄像头名字,调用【查询监控点列表v2…...

CSS实现服务卡片
CSS实现服务卡片 效果展示 CSS 知识点 回顾整体CSS知识点灵活运用CSS知识点 页面整体布局 <div class"container"><div class"card"><div class"box"><div class"icon"><ion-icon name"color-pal…...

问:如何判断系统环境是大端/小端存储?
大端存储(Big Endian)和小端存储(Little Endian)是两种不同的字节序(即字节顺序)规则,用于在计算机中存储和表示多字节数据类型(例如整数)。 概念解释 大端存储&#x…...

使用NumPy进行线性代数的快速指南
介绍 NumPy 是 Python 中用于数值计算的基础包。它提供了处理数组和矩阵的高效操作,这对于数据分析和科学计算至关重要。在本指南中,我们将探讨 NumPy 中可用的一些基本线性代数操作,展示如何通过运算符重载和内置函数执行这些操作。 元素级…...

uni-app之旅-day02-分类页面
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言创建cate分支4.1 渲染分类页面的基本结构4.2 获取分类数据4.3 动态渲染左侧的一级分类列表4.4 动态渲染右侧的二级分类列表4.5 动态渲染右侧的三级分类列表4.6 …...

鸿蒙harmonyos next flutter通信之BasicMessageChannel获取app版本号
本文将通过BasicMessageChannel获取app版本号,以此来演练BasicMessageChannel用法。 建立channel flutter代码: //建立通道 BasicMessageChannel basicMessageChannel BasicMessageChannel("com.xmg.basicMessageChannel",StringCodec());…...

【文件增量备份系统】MySQL百万量级数据量分页查询性能优化
🎯 导读:本文针对大数据量下的分页查询性能问题进行了深入探讨与优化,最初查询耗时长达12秒,通过避免全表计数及利用缓存保存总数的方式显著提升了浅分页查询速度。面对深分页时依然存在的延迟,采用先查询倒数第N条记录…...

音视频入门基础:FLV专题(12)——FFmpeg源码中,解析DOUBLE类型的ScriptDataValue的实现
一、引言 从《音视频入门基础:FLV专题(9)——Script Tag简介》中可以知道,根据《video_file_format_spec_v10_1.pdf》第80到81页,SCRIPTDATAVALUE类型由一个8位(1字节)的Type和一个ScriptDataV…...

【AI知识点】分层可导航小世界网络算法 HNSW(Hierarchical Navigable Small World)
HNSW(Hierarchical Navigable Small World)分层可导航小世界网络算法 是一种高效的近似最近邻搜索(Approximate Nearest Neighbor Search, ANN) 算法,特别适用于大规模、高维数据集的相似性检索。HNSW 基于小世界网络&…...

ubuntu图形界面右上角网络图标找回解决办法
问题现象: ubuntu图形界面右上角网络图标消失了,不方便联网: 正常应该是下图: 网络寻找解决方案,问题未解决,对于某些场景可能有用,引用过来: 参考方案 Ubuntu虚拟机没有网络图标或…...