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

🎈个人主页: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语言中的紧密关系。理解并掌握这些知识点,不仅有助于我们更加灵活地操作数组和指针,还能提升我们的编程能力和代码质量。希望本文对你有所帮助,祝你学习进步!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

我记不住的那些命令-xxdod

背景&#xff1a; 这里记录一下具体的xxd和od命令参数和使用方法&#xff0c;我想我肯定是记不住这些参数的。 零、文件 我们这里有一个示例文件README.txt&#xff0c;内容如下图所示&#xff1a; 一、xxd(查看、编辑二进制文件) 1. 十六进制显示(默认) 我们发现上面的显…

使用代理ip后访问网站仍然被拒该怎么办

目录 前言 一、代理IP被网站封锁 二、代理IP的质量不佳 三、代理设置不正确 总结 前言 在使用代理IP之后仍然被网站拒绝访问可能是由于多种原因引起的。这些原因包括代理IP被网站封锁、代理IP的质量不佳、代理设置不正确等。下面将详细介绍如何解决这些问题。 一、代理I…

代码学习记录16

随想录日记part16 t i m e &#xff1a; time&#xff1a; time&#xff1a; 2024.03.11 主要内容&#xff1a;今天的主要内容是二叉树的第五部分&#xff0c;主要涉及最大二叉树&#xff1b;合并二叉树&#xff1b;二叉搜索树的搜索&#xff1b;验证二叉搜索树。 654.最大二叉…

使用docker-compose部署Redis集群

一、部署三主三从的Redis集群 分别为6个节点建立挂载目录&#xff0c;每个目录下建立数据、配置、日志文件夹。 docker-compose内容如下&#xff1a; version: 3 services:redis1:image: redis:6.2.3restart: alwaysports:- "6379:6379"- "16379:16379"v…

Spring揭秘:ClassPathScanningProvider接口应用场景及实现原理!

技术应用场景 ClassPathScanningCandidateComponentProvider是Spring框架中一个非常核心的类&#xff0c;它主要用于在类路径下扫描并发现带有特定注解的组件&#xff0c;支持诸如ComponentScan、Component、Service、Repository和Controller等注解的自动扫描和注册。 ClassP…

.NET开源快速、强大、免费的电子表格组件

今天大姚给大家分享一个.NET开源&#xff08;MIT License&#xff09;、快速、强大、免费的电子表格组件&#xff0c;支持数据格式、冻结、大纲、公式计算、图表、脚本执行等。兼容 Excel 2007 (.xlsx) 格式&#xff0c;支持WinForm、WPF和Android平台&#xff1a;ReoGrid。 项…

普发Pfeiffer TPG256A MaxiGauge 真空计控制器接口通讯针脚等详情见图目录

普发Pfeiffer TPG256A MaxiGauge 真空计控制器接口通讯针脚等详情见图目录

强化学习中SARSA(State-Action-Reward-State-Action)和Q-learning的区别

SARSA&#xff08;State-Action-Reward-State-Action&#xff09;和Q-learning是两种经典的强化学习算法&#xff0c;它们都用于学习最优策略以使智能体在一个环境中获得最大的累积奖励。它们之间的主要区别在于它们更新动作值函数&#xff08;Q值函数&#xff09;的方式以及其…

SwiftUI组件-DatePicker

SwiftUI组件-DatePicker 本文记录一下SwiftUI组件-DatePicker import SwiftUIstruct DatePickerBootCamp: View {State var selectedDate: Date Date()var dateFormatter: DateFormatter {let formatter DateFormatter()formatter.dateStyle .shortformatter.timeStyle .…

使用kill()函数向进程发送信号

本片文章的学习记录总结来源于&#xff1a;https://www.bilibili.com/cheese/play/ep182660?csourcecommon_hp_history_null&t11&spm_id_from333.1007.top_right_bar_window_history.content.click 通常在Linux系统中&#xff0c;可以使用 kill or killall 命令向指定…

OpenCASCADE开发指南<十二>:OCC创建三维瓶子模型

在OpenCASCADE有一个例程&#xff0c;在 官方帮助网站中可以找到。程将教你如何使OpenCASCADE的API来进行三维建模。教程的目的不是描述所有的类&#xff0c;而是帮助你思考如何将OpenCASCADE作为一种工具。 1 概述 利用OpenCASCADE的API创建一个三维瓶子&#xff0c;形状如下…

如何在Linux部署DataEase数据分析服务并实现无公网IP远程分析内网数据信息

文章目录 前言1. 安装DataEase2. 本地访问测试3. 安装 cpolar内网穿透软件4. 配置DataEase公网访问地址5. 公网远程访问Data Ease6. 固定Data Ease公网地址 前言 DataEase 是开源的数据可视化分析工具&#xff0c;帮助用户快速分析数据并洞察业务趋势&#xff0c;从而实现业务…

IBM:《2024年消费者调研:无处不在的人工智能彻底变革零售业》

1月17日&#xff0c;IBM商业价值研究院最近发布了第三份两年一度的消费者调研报告。 这项名为《无处不在的人工智能彻底改变零售业&#xff1a;客户不会等待》的报告&#xff0c;对包含中国在内的全球近20000名消费者进行了调研&#xff0c;相关结果反映了消费者对零售体验的普…

【Python】进阶学习:一文了解NotImplementedError的作用

【Python】进阶学习&#xff1a;一文了解NotImplementedError的作用 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望…

2020-11-08 字符串指定位置倒序输出

缘由https://bbs.csdn.net/topics/398165677 //字符串指定位置倒序输出缘由https://bbs.csdn.net/topics/398165677char aa[] "abcABCabc\n";int a 3, b 5, c 0, d b;while (aa[c] ! \n)if (c < a || c > b)cout << aa[c]; else if(d > a) cout …

目标检测——YOLOv4算法解读

论文&#xff1a;YOLOv4&#xff1a;Optimal Speed and Accuracy of Object Detection 作者&#xff1a;Alexey Bochkovskiy, Chien-Yao Wang, Hong-Yuan Mark Liao 链接&#xff1a;https://arxiv.org/pdf/2004.10934.pdf 代码&#xff1a;https://github.com/AlexeyAB/darkne…

弹性盒子布局 Flexbox Layout

可以嵌套下去 1.display 属性 默认行排列 <style>.flex-item{ height: 20px;width: 10px;background-color: #f1f1f1;margin: 10px;}</style> </head> <body> <div class"flex-container"><div class"flex-item">1&l…

如何实现固定公网地址远程SSH连接Linux Deepin系统

文章目录 前言1. 开启SSH服务2. Deppin安装Cpolar3. 配置ssh公网地址4. 公网远程SSH连接5. 固定连接SSH公网地址6. SSH固定地址连接测试 前言 Deepin操作系统是一个基于Debian的Linux操作系统&#xff0c;专注于使用者对日常办公、学习、生活和娱乐的操作体验的极致&#xff0…

探索数据结构:双向链表的灵活优势

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;数据结构与算法 贝蒂的主页&#xff1a;Betty’s blog 1. 前言 前面我们学习了单链表&#xff0c;它解决了顺序表中插入删除需…

第110讲:Mycat实践指南:指定Hash算法分片下的水平分表详解

文章目录 1.应用指定Hash算法分片的概念2.使用应用指定Hash算法分片对某张表进行水平拆分2.1.在所有的分片节点中创建表结构2.2.配置Mycat实现应用指定Hash算法分片的水平分表2.2.1.配置Schema配置文件2.2.2.配置Rule分片规则配置文件2.2.3.配置Server配置文件2.2.4.重启Mycat …