C语言指针与数组(不适合初学者版):一篇文章带你深入了解指针与数组!

news2025/1/19 11:19:32

🎈个人主页:JAMES别扣了

💕在校大学生一枚。对IT有着极其浓厚的兴趣
✨系列专栏目前为C语言初阶、后续会更新c语言的学习方法以及c题目分享.
😍希望我的文章对大家有着不一样的帮助,欢迎大家关注我,我也会回关,大家一起交流一起互动,感谢大家的多多支持哈!

🎉欢迎 👍点赞✍评论⭐收藏

 前言

指针在C语言的版块中是有着极其重要的作用,通过两篇关于指针基础文章以及读者不断的代码实现可以很好掌握指针的基础知识,接下来我将结合C语言中数组方面的知识给读者们展示不一样的指针内容,这个系列文章我会写三篇左右,谢谢大家支持哈!

 往期相关内容高质量文章(93+质量分)

C语言数组:一篇文章让你秒懂基础!-CSDN博客

 C语言指针(上):一篇文章让你秒懂基础!-CSDN博客

 C语言指针(下):一篇文章让你秒懂基础!(5000字)-CSDN博客

1.数组名的理解

在之前的数组文章中,我们一起探讨过数组的基本知识,现在让我们更深一步理解数组的相关概念。

这⾥我们使⽤ &arr[0] 的⽅式拿到了数组第⼀个元素的地址,但是其实数组名本来就是地址,⽽且 是数组⾸元素的地址,我们来做个测试。

 #include <stdio.h>
 int main()
 {
 int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
 printf("&arr[0] = %p\n", &arr[0]);
 printf("arr      = %p\n", arr)

return 0;
}

输出结果:

我们发现数组名和数组⾸元素的地址打印出的结果⼀模⼀样,数组名就是数组⾸元素(第⼀个元素)的地 址。

这时候有读者会有疑问?数组名如果是数组⾸元素的地址,那下⾯的代码怎么理解呢?

#include <stdio.h>
 int main()
 {
 int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
 printf("%d\n", sizeof(arr));
 return 0;
 }

输出的结果是:40,如果arr是数组⾸元素的地址,那输出应该的应该是4/8才对。 其实数组名就是数组⾸元素(第⼀个元素)的地址是对的,但是有两个例外:

• sizeof(数组名),sizeof中单独放数组名,这⾥的数组名表⽰整个数组,计算的是整个数组的⼤⼩, 单位是字节

• &数组名,这⾥的数组名表⽰整个数组,取出的是整个数组的地址(整个数组的地址和数组⾸元素 的地址是有区别的)

除此之外,任何地⽅使⽤数组名,数组名都表⽰⾸元素的地址。

这时有好奇的同学,在试⼀下这个代码:

 #include <stdio.h>
 int main()
 {
 int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
 printf("&arr[0] = %p\n", &arr[0]);
 printf("arr     = %p\n", arr);
 printf("&arr    = %p\n", &arr);
 return 0;
)

哈哈,这时会有读者问那么arr与&arr有什么区别么?

 #include <stdio.h>
 int main()
 {
 int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
 printf("&arr[0]   = %p\n", &arr[0]);
 printf("&arr[0]+1 = %p\n", &arr[0]+1);
printf("arr       = %p\n", arr);
 printf("arr+1      = %p\n", arr+1);
printf("&arr       = %p\n", &arr);
printf("&arr+1     = %p\n", &arr+1);
return 0;
 }

输出结果:

&arr[0] = 0077F820

&arr[0]+1 = 0077F824

arr=0077F820

arr+1=0077F824

&arr=0077F820

&arr+1 = 0077F848

这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1相差4个字节,是因为&arr[0]和arr都是 ⾸元素的地址,+1就是跳过⼀个元素。 但是&arr和&arr+1相差40个字节,这就是因为&arr是数组的地址,+1操作是跳过整个数组的。 到这⾥⼤家应该搞清楚数组名的意义了吧。 

2. 使⽤指针访问数组

有了前⾯知识的⽀持,再结合数组的特点,我们就可以很⽅便的使⽤指针访问数组了。

#include <stdio.h>
 int main()
 {
 int arr[10] = {0}; //输⼊
 int i = 0;
 int sz = sizeof(arr)/sizeof(arr[0]); //输⼊
 
int* p = arr;
 for(i=0; i<sz; i++)
 {
 scanf("%d", p+i);
 //scanf("%d", arr+i);//也可以这样写
 
} //输出
 
for(i=0; i<sz; i++)
 {
 printf("%d ", *(p+i));
 }
 return 0;
 }

这个代码搞明⽩后,我们再试⼀下,如果我们再分析⼀下,数组名arr是数组⾸元素的地址,可以赋值 给p,其实数组名arr和p在这⾥是等价的。那我们可以使⽤arr[i]可以访问数组的元素,那p[i]是否也可 以访问数组呢?

#include <stdio.h>
 int main()
 {
 int arr[10] = {0}; //输⼊
 
int i = 0;
 int sz = sizeof(arr)/sizeof(arr[0]); //输⼊
 
int* p = arr;
 for(i=0; i<sz; i++)
 {
 scanf("%d", p+i);
 //scanf("%d", arr+i);//也可以这样写
 
} //输出
 
for(i=0; i<sz; i++)
 {
 printf("%d ", p[i]);
 }
 return 0;
 }

将*(p+i)换成p[i]也是能够正常打印的,所以本质上p[i]是等价于*(p+i)。

同理arr[i] 应该等价于*(arr+i),数组元素的访问在编译器处理的时候,也是转换成⾸元素的地址+偏移 量求出元素的地址,然后解引⽤来访问的。

3. ⼀维数组传参的本质

数组我们学过了,之前也讲了,数组是可以传递给函数的,这个⼩节我们讨论⼀下数组传参的本质。 ⾸先从⼀个问题开始,我们之前都是在函数外部计算数组的元素个数,那我们可以把函数传给⼀个函 数后,函数内部求数组的元素个数吗?

#include <stdio.h>
 void test(int arr[])
 {
 int sz2 = sizeof(arr)/sizeof(arr[0]);
 printf("sz2 = %d\n", sz2);
 }
 int main()
 {
 int arr[10] = {1,2,3,4,5,6,7,8,9,10};
 int sz1 = sizeof(arr)/sizeof(arr[0]);
 printf("sz1 = %d\n", sz1);
 test(arr);
 return 0;
 }

输出的结果:

我们发现在函数内部是没有正确获得数组的元素个数。

这就要学习数组传参的本质了,上个⼩节我们学习了:数组名是数组⾸元素的地址;那么在数组传参 的时候,传递的是数组名,也就是说本质上数组传参本质上传递的是数组⾸元素的地址。 所以函数形参的部分理论上应该使⽤指针变量来接收⾸元素的地址。那么在函数内部我们写 sizeof(arr) 计算的是⼀个地址的⼤⼩(单位字节)⽽不是数组的⼤⼩(单位字节)。正是因为函 数的参数部分是本质是指针,所以在函数内部是没办法求的数组元素个数的。

void test(int arr[])//参数写成数组形式,本质上还是指针
 printf("%d\n", sizeof(arr));

 }
 void test(int* arr)//参数写成指针形式
 
{
 printf("%d\n", sizeof(arr));//计算⼀个指针变量的⼤⼩
 
}
 int main()
 {
 int arr[10] = {1,2,3,4,5,6,7,8,9,10};
 test(arr);
 return 0;
 }
 
{

结论:⼀维数组传参,形参的部分可以写成数组的形式,也可以写成指针的形式。

4.⼆级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪⾥?答案是:更高级别的指针!

这就是 ⼆级指针。

对于⼆级指针的运算有:

1.*ppa 通过对ppa中的地址进⾏解引⽤,这样找到的是 

int b = 20;

*ppa = &b;// 等价于 pa = &b; pa , *ppa 其实访问的就是 pa .

2.**ppa 先通过 *ppa 找到 pa ,然后对pa进行解应用操作:*pa,那找到的是a。

**ppa = 30;

// 等价于 *pa = 30;

// 等价于 a = 30;

5. 指针数组

指针数组是指针还是数组? 我们类⽐⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。 那指针数组呢?是存放指针的数组.

 指针数组的每个元素都是⽤来存放地址(指针)的。 如下图:

指针数组的每个元素是地址,⼜可以指向⼀块区域.

6. 指针数组模拟⼆维数组

#include <stdio.h>
 int main()
 {
 int arr1[] = {1,2,3,4,5};
 int arr2[] = {2,3,4,5,6};
 int arr3[] = {3,4,5,6,7};
 //数组名是数组⾸元素的地址,类型是int*的,就可以存放在parr数组中
 
int* parr[3] = {arr1, arr2, arr3};
 int i = 0;
 int j = 0;
 for(i=0; i<3; i++)
 {
 for(j=0; j<5; j++)
 {
 printf("%d ", parr[i][j]);
 }
 printf("\n");
 }
return 0;
 }

 parr[i]是访问parr数组的元素,parr[i]找到的数组元素指向了整型⼀维数组,parr[i][j]就是整型⼀维数 组中的元素。 上述的代码模拟出⼆维数组的效果,实际上并⾮完全是⼆维数组,因为每⼀⾏并⾮是连续的。

总结

C语言中数组与指针的深入探索

在C语言编程中,数组和指针是两个紧密相关的概念。数组是一种数据结构,用于存储相同类型的元素集合;而指针则是一种变量,用于存储另一个变量的内存地址。理解数组和指针之间的关系,对于深入掌握C语言至关重要。

一、数组名的理解

在C语言中,数组名实际上是一个指向数组首元素的常量指针。这意味着数组名本身并不表示整个数组,而是指向数组第一个元素的地址。因此,我们不能对数组名进行赋值操作,但可以获取其地址或进行解引用操作。

二、使用指针访问数组

通过指针,我们可以直接访问数组中的元素。假设我们有一个整型数组int arr[5];,我们可以使用指针int *p = arr;来指向数组的首元素。此时,*p就表示数组的第一个元素,*(p+1)表示第二个元素,以此类推。通过这种方式,我们可以遍历整个数组。

三、一维数组传参的本质

在C语言中,当我们将一维数组作为参数传递给函数时,实际上传递的是数组首元素的地址。因此,函数内部接收到的参数实际上是一个指针,指向数组的首元素。这使得我们可以在函数内部通过指针来访问和修改数组的元素。

四、二级指针

二级指针是指向指针的指针。它的声明形式为int **pp;,其中pp是一个指向整型指针的指针。二级指针常用于动态内存分配、函数返回多个值以及处理指针数组等场景。

五、指针数组

指针数组是指数组的每个元素都是指针。例如,int *pArr[5];表示一个包含5个整型指针的数组。每个指针可以指向一个整型变量或整型数组的首元素。指针数组常用于存储多个字符串的地址或处理多个动态分配的数组。

六、指针数组模拟二维数组

在C语言中,二维数组在内存中是连续存储的,但我们可以使用指针数组来模拟二维数组的行为。通过为每个指针分配一个一维数组,我们可以创建一个不规则的“二维数组”。这种方式在处理稀疏矩阵或动态调整数组大小时非常有用。

通过上述内容的阐述,我们可以看到数组和指针在C语言中的紧密关系。理解并掌握这些知识点,不仅有助于我们更加灵活地操作数组和指针,还能提升我们的编程能力和代码质量。希望本文对你有所帮助,祝你学习进步!

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

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

相关文章

python 基础知识点(蓝桥杯python科目个人复习计划63)

今日复习内容&#xff1a;做题 例题1&#xff1a;蓝桥骑士 问题描述&#xff1a; 小蓝是蓝桥王国的骑士&#xff0c;他喜欢不断突破自我。 这天蓝桥国王给他安排了N个对手&#xff0c;他们的战力值分别为a1,a2,...,an&#xff0c;且按顺序阻挡在小蓝的前方。对于这些对手小…

Java NIO浅析

NIO&#xff08;Non-blocking I/O&#xff0c;在Java领域&#xff0c;也称为New I/O&#xff09;&#xff0c;是一种同步非阻塞的I/O模型&#xff0c;也是I/O多路复用的基础&#xff0c;已经被越来越多地应用到大型应用服务器&#xff0c;成为解决高并发与大量连接、I/O处理问题…

矿洞隧道漫游可视化:探索地心深处的奇幻世界

在这个充满好奇与探索的时代&#xff0c;我们总是渴望揭开世界的神秘面纱&#xff0c;探寻那些深藏在地球内部的奥秘。 矿洞隧道漫游可视化系统通过先进的计算机图形学、虚拟现实和三维建模技术&#xff0c;将矿洞隧道的真实场景进行高精度还原&#xff0c;让我们仿佛置身于一个…

leetcode判断子序列

本题中&#xff0c;我们可以删除原始字符串的一些字符但是不能改变其他字符的位置&#xff0c;这种求子序列的题都可以用动态规划来解决。 首先我们要确定dp数组的定义&#xff0c;这里我们将dp数组定义为dp[i][j] 表示以下标i-1为结尾的字符串s&#xff0c;和以下标j-1为结尾的…

Cadence PCB布线时的电源地操作

一般在布线时需要先将GND、3.3V、5V、24V电源和地等进行处理&#xff0c;使得在打开飞线后不至于太凌乱。 效果如下&#xff1a; 1. 方法如下&#xff1a; 之后打开飞线即可显示设置效果。 2. 至于如何删除&#xff1f;&#xff1f;&#xff1f; 勾选delete&#xff0c;应用即…

如何选择多域名SSL证书?

多域名SSL证书&#xff0c;顾名思义&#xff0c;是一种特殊的SSL证书&#xff0c;它允许同时保护多个域名及其子域名的安全。与传统的单域名SSL证书相比&#xff0c;多域名SSL证书就像是一把万能钥匙&#xff0c;能够开启多个不同的锁&#xff0c;为多个网站提供一把坚固的安全…

【机器学习】机器学习是什么?用在哪里?怎么用?

1.机器学习是什么&#xff1f; 机器学习&#xff08;Machine Learning&#xff09;是人工智能的一个分支&#xff0c;它是一种通过对数据进行训练和学习&#xff0c;让计算机系统从中获取知识并改善性能的方法。简而言之&#xff0c;机器学习使计算机具有从数据中学习并自动改…

垃圾分类网站 |基于springboot框架+ Mysql+Java+B/S结构的垃圾分类网站 设计与实现(可运行源码+数据库+设计文档+部署说明)

推荐阅读100套最新项目 最新ssmjava项目文档视频演示可运行源码分享 最新jspjava项目文档视频演示可运行源码分享 最新Spring Boot项目文档视频演示可运行源码分享 目录 前台功能效果图 管理员功能登录前台功能效果图 系统功能设计 数据库E-R图设计 lunwen参考 摘要 研究…

重生奇迹mu元素合成流程

1、元素之魂是通过元素之心和艾丽亚之环合成的&#xff0c;在NPC阿德尼这里可以合成&#xff0c;NPC在精灵地图&#xff0c;而元素之心碎片和艾丽亚之环碎片的掉落也是在是在精灵地图。 2、玩家获得材料后&#xff0c;需经过2阶段的精炼才会成为可使用的元素之魂。各精炼阶段的…

在线动漫信息平台|基于springboot框架+ Mysql+Java+B/S架构的在线动漫信息平台设计与实现(可运行源码+数据库+设计文档)

推荐阅读100套最新项目 最新ssmjava项目文档视频演示可运行源码分享 最新jspjava项目文档视频演示可运行源码分享 最新Spring Boot项目文档视频演示可运行源码分享 目录 前台功能效果图 会员后台功能 管理员功能登录前台功能效果图 系统功能设计 数据库E-R图设计 lunwen…

Docker 哲学 - 容器操作

容器&#xff1a; 创建 停止 删除 强制删除&#xff08;正在运行&#xff09; run stop rm rm -f 列出本地容器&#xff1a; docker ps / docker container ls 镜像&#xff1a; search pull run &#xff1a; …

MySQL Connector连接失败之SSL connection error: protocol version mismatch

调用 mysql_real_connect&#xff08;&#xff09; 连接失败&#xff0c;报错为ERROR 2026 (HY000): SSL connection error: protocol version mismatch 调用mysql_error&#xff08;&#xff09;查看失败原因&#xff0c;结果为 SSL connection error: protocol version …

Bugku MISC做题笔记

简单套娃DX 这一题需要对png图片的结构有所了解。详细可参考https://www.w3.org/TR/png/ 幸好每一张图片只有一个错误&#xff0c;逐步调试&#xff0c;就可以发现所有错误&#xff0c;修正即可。具体错误参看python程序中的注释&#xff1a; import ossrc_dir .\\XD\\ de…

7-Eleven用工数字化:零售哲学下的人效管理实践

2014年&#xff0c;一本《零售的哲学》在中国掀起热潮&#xff0c;揭示了7-Eleven便利店的新零售坪效管理秘诀。而对大部分零售企业来说&#xff0c;劳动力效率是坪效背后的主要支柱。近期&#xff0c;国内领先的劳动力管理云服务提供商盖雅工场发布了《聚焦人效、重塑组织&…

香港理工大学主办!2024年第八届电力能源系统与应用国际会议(ICoPESA 2024)即将召开!

2024年第八届电力能源系统与应用国际会议&#xff08;ICoPESA 2024&#xff09; 2024年6月24日-26日 中国香港 ICoPESA 2024-Hong Kong (icpesa.org)https://icpesa.org/index.html 会议组织单位 会议出版及检索&#xff1a; 会议录用并注册的论文将由IEEE出版&#xff0c;…

聚酰亚胺PI材料难于粘接,用什么胶水粘接?那么让我们先一步步的从认识它开始(一)

聚酰亚胺PI的基本概念 聚酰亚胺&#xff08;Polyimide&#xff0c;简称PI&#xff09;是一种重要的高性能聚合物材料。是指主链上含有酰亚胺环的一类聚合物&#xff0c;是综合性能最佳的有机高分子材料之一。它具有最高的阻燃等级&#xff08;UL-94&#xff09;&#xff0c;以及…

OpenCV 将rgb图像转化成字符图像

将RGB图像转换成字符图像&#xff08;ASCII art&#xff09;通常涉及到灰度化、降采样、映射字符等一系列步骤。以下是一个简化的OpenCVC实现示例&#xff1a; #include <opencv2/opencv.hpp> #include <iostream> #include <string>// 字符映射表&#xff…

MySQL—redo log、undo log以及MVCC

MySQL—redo log、undo log以及MVCC 首先回忆一下MySQL事务的四大特性&#xff1a;ACID&#xff0c;即原子性、一致性、隔离性和持久性。其中原子性、一致性、持久性实际上是由InnoDB中的两份日志保证的&#xff0c;一份是redo log日志&#xff0c;一份是undo log日志&#xff…

国产Copilot--通义灵码安装教程

文章目录 在 Visual Studio Code 中安装通义灵码步骤1步骤2步骤3步骤4 参考 在 Visual Studio Code 中安装通义灵码 通义灵码&#xff0c;是一款基于通义大模型的智能编码辅助工具&#xff0c;提供行级/函数级实时续写、自然语言生成代码、单元测试生成、代码注释生成、代码解…

LLM - 大语言模型(LLM) 概述

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://blog.csdn.net/caroline_wendy/article/details/136617643 大语言模型(LLM, Large Language Model)的发展和应用是一个非常广泛的领域&#xff0c;涉及从早期的统计模型到现代基于深度学…