【C语言】—— 指针三 : 参透数组传参的本质

【C语言】—— 指针三 : 参透数组传参的本质

    • 一、数组名的理解
    • 二、使用指针访问数组
      • 2.1、指针访问数组
      • 2.2、[ ] 的深入理解
      • 2.3、数组与指针的区别
    • 三、一维数组的传参本质
    • 四、数组指针变量
      • 4.1、数组指针变量是什么
      • 4.2、 数组指针的初始化
    • 五、二维数组传参的本质

一、数组名的理解

  
  如果我们想用指针来访问数组,我们可以怎么做呢?
  

int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
int* p = &arr[0];

  
  之前我们都是通过 & a r r arr 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[0] + 1  = %p\n", &arr[0] + 1);
	printf("arr          = %p\n", arr);
	printf("arr + 1      = %p\n", arr + 1);
	return 0;
}

  
输出结果:
在这里插入图片描述

  
  我们发现,他们的输出结果是一样的,进行指针运算,同样是跳过了四个字节,所以数组名就是数组首元素的地址
  
  这时,有小伙伴就要问了,那我之前用 s i z e o f sizeof sizeof 为什么能求出整个数组的长度呢
  

#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,如果是地址的话,输出的结果应该是 4/8,为什么呢?难道前面说错了?
  
  其实数组名是地址这本身并没有错,但是有两个例外:
  

  • s i z e o f (数组名) sizeof(数组名) sizeof(数组名) s i z e o f 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[0]+1 = %p\n", &arr[0]);
	printf("arr       = %p\n", arr);
	printf("arr+1     = %p\n", arr);
	printf("&arr      = %p\n", &arr);
	printf("&arr+1    = %p\n", &arr);

	return 0;
}

  
输出结果:
在这里插入图片描述

  我们可以看到,&arr[0]&arr[0]+1 相差 4 个字节,同样 a r r arr arr a r r + 1 arr+1 arr+1 也是相差 4 个字节
  
  & a r r arr arr& a r r arr arr[0] a r r arr arr 在数值上相等,但 & a r r arr arr& a r r + 1 arr+1 arr+1 相差 40 个字节,这是因为 & a r r arr arr 取出的是整个数组的地址,+1 跳过的是整个数组,而该数组的大小为 40 个字节
  
  看到这,相信大家对数组名有了更深层次的理解了吧。
  
  

二、使用指针访问数组

2.1、指针访问数组

  
  学了前面的知识,我们就可以很方便地使用指针访问数组了:

#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 p p,这里 p p p a r r arr arr 是等价的,我们可以用 a r r arr arr [ i i i ] 来访问数组,那可不可以用 p p p [ i i 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 p p [ i i i ] 等价于 ∗ ( p + i ) *(p+i) p+i,同理, a r r arr arr [ i i i ] 等价于 ∗ ( a r r + 1 ) *(arr+1) arr+1
  

2.2、[ ] 的深入理解

  
  这里,我们对下标引用操作符进行更深层次的理解
  
   ∗ ( a r r + 1 ) *(arr+1) arr+1 由加法交换律,应该与 ∗ ( i + a r r ) *(i+arr) i+arr 等价,事实上也确实如此。这时,我们不妨做一个大胆的猜想:

  • 既然 ∗ ( a r r + i ) *(arr+i) arr+i等价于 a r r arr arr [ i i i ],那 ∗ ( i + a r r ) *(i+arr) i+arr 是否等价于 i i i [ a r r arr arr ] 呢?
  • 那这样的话, a r r arr arr [ i i i ] 是不是与 i i i [ a r r arr arr ] 相等呢 ?那他们四种表示是不是相等呢?
      

在这里插入图片描述

  
实践出真知,我们直接上代码试试:

#include<stdio.h>

int main()
{
	int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	int i = 0;

	//arr[i]打印
	for (i = 0; i < sz; i++)
	{
		printf("%d ", arr[i]);
	}
	printf("\n");

	//*(arr + i)打印
	for (i = 0; i < sz; i++)
	{
		printf("%d ", *(arr + i));
	}
	printf("\n");

	//*(i + arr)打印
	for (i = 0; i < sz; i++)
	{
		printf("%d ", *(i + arr));
	}
	printf("\n");

	//i[arr]打印
	for (i = 0; i < sz; i++)
	{
		printf("%d ", i[arr]);
	}
	printf("\n");
	return 0;
}

运行结果:
在这里插入图片描述

  

在这里插入图片描述

  
  可以看到,猜想是 正确 的,其实下标引用操作符 [ ] 并没有那么神秘,它仅仅只是一个操作符而已,就像 + 操作符一样,它的两个操作数可以互换位置,[] 也是如此,事实上, a r r arr arr [ i i i ] 在程序运行时,编译器都会以 ∗ ( a r r + i ) *(arr+i) arr+i 的形式执行。当然,还是尽量别写成 i i i [ a r r arr arr ] 的形式,这样代码可读性不高
  

2.3、数组与指针的区别

  
  讲了这么多,大家可能对数组与指针的区别有点糊涂了吧,这里我们来理一下

  • 数组就是数组,是一块连续的空间。数组的大小和数组元素类型和数组大小都有关系。
  • 指针(变量)就是指针(变量),大小 4/8 字节。
  • 联系:数组名是地址,是首元素地址,可使用指针访问数组

  
  

三、一维数组的传参本质

  
  在【C语言】—— 指针二 : 初识指针(下)一文中,我曾经提到,将变量名传递给函数,是传值调用形参仅仅是实参的一份拷贝,无法通过函数改变实参的值。
  
  而将数组名传给函数,却可以通过函数改变数组的值。这是为什么呢,想到之前学过的传址调用(详情请看【C语言】—— 指针二 : 初识指针(下)),再结合刚刚所学到的知识:数组名是函数首元素的地址
  
  答案就不言而喻了:数组传参,传递的是数组名本质上传递的是数组首元素的地址
  
  同时,形参中创建的数组不会再单独创建数组空间的,所以形参是可以省略数组大小

  这时,我们不妨想一个问题:之前我们都是在函数外面将数组的元素个数求好,再传递给函数,我们可以把数组传递给函数后,在函数内部求数组的元素个数吗
  

#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;
}

  
输出结果:
在这里插入图片描述

  可见,函数内部并没有正确求出数组的元素个数
  
  为什么呢?因为数组传参时,传递的是数组首元素的指针,虽然函数中 s i z e o f ( a r r ) sizeof(arr) sizeofarr a r r arr arr 是单独放在 s i z e o f sizeof sizeof 中的,但前面函数传参时, a r r arr arr 为指针,函数将 a r r arr arr 当做指针变量来接收,所以这里的 a r r arr arr 也被认为是指针,大小为 4/8,所以最终结果为 1/2。
  
  所以在函数内部是无法求出数组元素个数的
  

#include<stdio.h>

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、数组指针变量是什么

  
  数组指针变量是什么,之前我们学过指针数组(详情请看【C语言】—— 指针二 : 初识指针(下)),指针数组是一种数组,存放的是指针变量。
  
  那么数组指针呢?是数组还是变量?
  
  前面,我们学习过:整形指针变量 i n t int int ∗ * ,存放的是整形变量的地址,能够指向整形数据的指针字符指针变量 c h a r char char ∗ * ,存放字符形变量的地址,能够指向字符型数据的指针。
  
所以,数组指针变量应该是: 存放的是数组的地址,能够指向数组的指针变量
  
下面代码哪个是数组指针变量:

int* p1[10];
int (*p2)[10];

  分析之前,我们先要知道: [ ] 的优先级比 ∗ *

  第一个, p p p 先与 [ ] 结合,表示是一个数组,剩下的 i n t int int ∗ * 表示数组存放的元素是 i n t int int ∗ * ,为指针数组

  第二个,由于有(), p p p 先与 ∗ * 结合,表示是一个指针变量,然后指针指向的是大小为 10 个整型的数组。 p p p 是一个指针,指向一个数组,为数组指针变量
  

4.2、 数组指针的初始化


  数组指针变量是用来存放数组的地址的,那怎么获得数组的地址呢,这时就需要用到前面学的 &数组名
  

int arr[10] = { 10 };
&arr;//得到的是数组的地址

  
  如果要存放数组的地址,就得存放在数组指针变量中
  

int(*p)[10] = &arr;

  
在这里插入图片描述

注:下标引用操作符 [ ] 中 10 不能省略,同时,要匹配得上。
  
  
数组指针类型解析:

在这里插入图片描述

  
  

五、二维数组传参的本质

  
  有了前面对数组指针的了解,我们就可以了解二维数组传参的本质了。

  过去,我们想传递二维数组时,可以这样写
  

#include<stdio.h>

void test(int a[3][5], int r, int c)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < r; i++)
	{
		for (j = 0; j < c; j++)
		{
			printf("%d ", a[i][j]);
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };
	test(arr, 3, 5);
	return 0;
}

  这里,实参是二维数组,形参也是二维数组的形式,那还有什么其他方法吗。
  
  我们可以类比一下一维数组。对一维数组来说,除了可以传递一维数组的形式,还可以传递数组名,即首元素地址,那二维数组可不可以也传递数组名呢?
  
  答案是 肯定 的,二维数组的数组名也是首元素的地址,但这之前,我们需再次理解二维数组:
  
  二维数组可以看做每个元素是一维数组的数组,即二维数组每个元素是一维数组,那么二维数组的首元素就是第一行,即第一个一维数组。
  
  所以,根据数组名就是数组首元素的地址这个规则,二维数组数组名是第一个一维数组的地址,其类型为 i n t [ 5 ] int[5] int[5] 。这样,在函数接收参数时,接收的是数组地址,所以形参应该为数组指针变量,类型为 i n t ( ∗ ) [ 5 ] int(*)[5] int[5].
  
  二维数组传参本质上也是传递地址,传递的是第一行这个一维数组的地址
  

#include<stdio.h>

void test(int(*p)[5], int r, int c)
{
	int i = 0;
	int j = 0;
	for (i = 0; i < r; i++)
	{
		for (j = 0; j < c; j++)
		{
			printf("%d ", *(*(p + i) + j));
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };
	test(arr, 3, 5);
	return 0;
}

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

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

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

相关文章

【LabVIEW FPGA入门】插值、输出线性波形

概述 NI 的可重配置 I/O (RIO) 硬件使开发人员能够创建自定义硬件&#xff0c;以在坚固耐用、高性能和模块化架构中执行许多任务&#xff0c;而无需了解低级 EDA 工具或硬件设计。使用 RIO 硬件轻松实现的此类任务之一是模拟波形生成。本教程介绍了使用 CompactRIO 硬件和 LabV…

计算机网络:计算机网络概述

计算机网络&#xff1a;计算机网络概述 因特网概述网络&#xff0c;互连网&#xff0c;因特网因特网发展的三个阶段因特网的标准化工作因特网组成 计算机网络的定义计算机网络的分类按使用者分类按传输介质分类按网络的覆盖范围分类按拓扑结构分类 因特网概述 网络&#xff0c…

投简历没回复?9位DBA公众号集结,快上车!

&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&#x1f61c;&#x1f61c; 中国DBA联盟(ACD…

Unicode转码 [ASIS 2019]Unicorn shop1

打开题目 我们买最贵的试试看&#xff0c;结果提示只能输入一个字符 抓包分析一下看看 从中可以发现源代码是如何处理price的 使用的是unicodedata.numeric() 但是我们查看页面源代码&#xff0c;发现页面的编码是utf-8编码 所以&#xff0c;前端html使用的是utf-8&#xff0…

【学习】CMMI评估认证的意义和需要注意的问题

​ CMMI认证是软件能力成熟度集成模型&#xff0c;是软件行业中的一种质量管理体系&#xff0c;旨在评估软件开发组织的成熟度和能力&#xff0c;以帮助企业提高软件质量、降低成本、控制风险&#xff0c;并获得更好的商业效益。 一、CMMI评估认证的意义 1. 提高软件质量&am…

win提权第二弹服务提权

阅读须知&#xff1a; 探索者安全团队技术文章仅供参考,未经授权请勿利用文章中的技术资料对任何计算机系统进行入侵操作,由于传播、利用本公众号所提供的技术和信息而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者 本人负责&#xff0c;作者不为此承担任何责任,如…

Windows 10中打开控制面板的13种方法,总有一种适合你

前言 虽然有传言称微软将取消控制面板,但它不会那么快消失。一些重要的设置仅在Windows 10的经典控制面板中找得到,它们不在设置应用程序中。本文有13种方法可以打开控制面板。 搜索开始菜单 你可以使用“开始”菜单的搜索功能搜索PC上的任何应用程序。在任务栏左侧的搜索…

基于微信小程序的电影交流平台

技术&#xff1a;springbootmysqlvue 一、背景 互联网发展至今&#xff0c;无论是其理论还是技术都已经成熟&#xff0c;而且它广泛参与在社会中的方方面面。它让信息都可以通过网络传播&#xff0c;搭配信息管理工具可以很好地为人们提供服务。所以各行业&#xff0c;尤其是规…

[蓝桥杯 2015 省 B] 生命之树

水一水的入门树形DP #include<iostream> #include<algorithm> #include<vector> using namespace std; using ll long long; #define int long long const int N 2e610; const int inf 0x3f3f3f3f; const int mod 1e97;int n; int w[N]; vector<vecto…

环境检测LIMS系统 环境检测实验室信息管理系统

环境检测行业在所有检测领域流程最长&#xff0c;数据量最大&#xff0c;专家组不同&#xff0c;认证体系的记录单/报告模板也是各自不同&#xff0c;因此如何选择一套适用本企业的LIMS也成为重中之重的工作&#xff0c;好的系统可以给企业带来非常大的便捷&#xff0c;也能大大…

4 Redis持久化

Redis 是一个内存数据库&#xff0c;所以其运行效率非常高。但也存在一个问题&#xff1a;内存中的数据是不持久的&#xff0c;若主机宕机或 Redis 关机重启&#xff0c;则内存中的数据全部丢失。当然&#xff0c;这是不允许的。Redis 具有持久化功能&#xff0c;其会按照设置以…

让AI给你写代码(五)—— 应用Agent,理解Agent,走进现实世界

本文想解决一个问题&#xff0c;理解Agent有啥具体的作用&#xff1f; 所谓读书千遍&#xff0c;不如动手一试&#xff0c;我们还是借助于上一篇&#xff0c;让AI给你写代码&#xff08;四&#xff09;—— 初步利用LangChain Agent根据输入生成&#xff0c;保存&#xff0c;执…

基于springboot的药房进销存管理系统

光明医院药品库房信息管理系统的设计与实现医院管理员 1.医院管理员信息管理—增加删除修改人员信息(人员信息包括年龄性别学历) 2医院管理员账号密码修改 3发布公告—医院管理者发布医院公告对药库管理者可见 4查看药品入库出库信息 (药品厂商&#xff0c;生产日期&#xff0c…

【接口测试】神器JMeter

‍1 JMeter是什么 Apache JMeter是Apache组织开发的一款开源软件&#xff0c;是一款非常好用的接口测试工具。它的特点是开源免费&#xff0c;简单好用。 我们在测试过程需要做接口测试的话就可以使用它&#xff0c;也可以用来批量造数据&#xff0c;接下来我们就来看看JMete…

统计学第1天

描述性统计 统计数据类型分类 按计量尺度划分 分类数据 能归于某一类别的非数字数据&#xff0c;数据是对事进行分类的结果&#xff0c;结果表现为类别&#xff0c;用文字来描述。 例如&#xff1a;人口按照性别&#xff08;男、女&#xff09;&#xff0c;企业按照所处行业…

进程(2)——进程优先级

1、基本概念 cpu资源分配的先后顺序&#xff0c;就是指进程的优先权&#xff08;priority&#xff09;。 优先权高的进程有优先执行权利。配置进程优先权对多任务环境的linux很有用&#xff0c;可以改善系统性能。 还可以把进程运行到指定的CPU上&#xff0c;这样一来&a…

【Unity】捕捉PC桌面的插件

【背景】 之前介绍了如何用一款名为uWindowCapture的Unity免费插件在Unity的Canvas上展示PC桌面。经过一段时间的使用,本篇继续分享此插件的一些功能和限制。 在此感谢作者Hecomi。 【特征和限制】 一般局域网络环境只能最多达到15帧的帧率,所以别幻想用来窜流游戏或者看电…

Linux环境开发工具之vim

前言 上一期我们已经介绍了软件包管理器yum&#xff0c; 已经可以在linux上查找、安装、卸载软件了&#xff0c;本期我们来介绍一下文本编辑器vim。 本期内容介绍 什么是vim vim的常见的模式以及切换 vim命令模式常见的操作 vim底行模式常见的操作 解决普通用户无法执行sudo问…

【机器学习300问】40、如何评估一个异常检测系统?

上一篇文章是我学习异常检测系统如何实现的学习笔记&#xff0c;这篇文章接着上文记录几个评价异常检测系统的关键步骤和指标。如果友友们没有看过之前的文章可以点击下面的链接去看看哦&#xff01; 【机器学习300问】39、高斯分布模型如何实现异常检测&#xff1f;http://t.…

【鸿蒙HarmonyOS开发笔记】通知模块之发布基础类型通知,内含如何将图片变成PixelMap对象

通知简介 应用可以通过通知接口发送通知消息&#xff0c;终端用户可以通过通知栏查看通知内容&#xff0c;也可以点击通知来打开应用。 通知常见的使用场景&#xff1a; 显示接收到的短消息、即时消息等。 显示应用的推送消息&#xff0c;如广告、版本更新等。 显示当前正…
最新文章