C语言 用数组名作函数参数
当用数组名作函数参数时,如果形参数组中各元素的值发生变化,实参数组元素的值随之变化。
1.数组元素做实参的情况:
如果已经定义一个函数,其原型为
void swap(int x,int y);
假设函数的作用是将两个形参(x,y)进行交换,现在有以下的函数调用:
swap (a[0],a[1]);
用数组元素做实参的情况,与用变量作实参的情况一样,是“值传递”方式,将a[0]和a[1]的值单向传递给x和y。当x和y的值改变时a[0]和a[1]的值并不改变。
2.数组名作函数形参的情况:
实参数组名代表该数组首元素的地址,而形参是用来接收从实参传递过来的数组首元素的地址的。因此,形参应该是一个指针变量(只有指针变量才能存放地址)。实际上,C编译都是将形参数组名作为指针变量来处理的。
例如:定义一个函数fun,fun函数的形参写成数组的形式:
void fun(int arr[],int n);
但在程序编译时是将arr按指针变量来处理的,相当于函数fun应写成:
void fun(int *arr,int n);
在该函数被调用时,系统会在fun函数中建立一个指针变量arr,用来存放从主调函数传递过来的实参数组元素的地址。如果在fun函数中用运算符sizeof测定arr所占字节数,可以发现sizeof(arr)的值为4。这就证明了系统是把arr作为指针变量来处理的。例如:
void fun(int arr[], int n)
{int s = sizeof(arr);printf("arr所占字节数为:%d\n", s);
}
int main()
{int brr[10];fun(brr, 10);return 0;}
运行结果:
当arr接收了实参数组的首元素地址后,arr就指向实参数组的首元素,也就是指向了brr[0]。因此,arr就是brr[0]。arr+1指向brr[1],arr+2指向brr[2],arr+3指向brr[3].也就是说,*(arr+1)
,*(arr+2)
,*(arr+3)
分别是brr[1],brr[2],brr[3]。*(arr+i)
和arr[i]
是无条件等价的。因此在调用函数期间,arr[0]和arr以及brr[0]都代表数组brr序号为0的元素。
【注意】
实参数组名代表一个固定的地址,或者说是指针常量,但形参数组名并不是一个固定的地址,而是按指针变量处理。
在函数调用进行虚实结合后,形参的值就是实参数组首元素的地址。在函数指向期间,它可以在被赋值。
【例】定义一个数组名作形参的函数,通过调用这个函数改变实参数组的值
int fun(int arr[], int n)
{//int s = sizeof(arr);//printf("arr所占字节数为:%d\n", s);int t;t = arr[1];arr[1] = arr[2];arr[2] = t;return arr[0];
}
int main()
{int brr[10] = { 1,3,2,4,5,6,7,8,9,10 };fun(brr, 10);for (int i = 0; i < 10; i++){printf("%d ", brr[i]);}return 0;}
运行结果:
常用这种方法通过调用一个函数来改变实参数组的值。
3.变量名做函数参数和用数组名做函数参数的比较
(1)当实参类型是变量名时,要求形参的类型也是变量名,通过形参传递的信息是变量的值,通过函数调用不能改变实参变量的值。
(2)当实参类型是数组名时,要求形参的类型是数组名或者指针变量,通过形参传递的信息是实参数组首元素的地址,通过函数调用能改变实参变量的值。
说明:C语言调用函数时虚实结合的方法都是采用“值传递”方式,当用变量名作为函数参数时传递的是变量的值。当用数组名作函数参数时,由于数组名代表的是数组首元素地址,因此传递的值是地址,所以要求形参为指针变量。
4.数组名和指针变量作为函数的形参
在C语言中用下标法和指针法都可以访问一个数组(如果有一个数组a,则a[i]和*(a+i)无条件等价)。用数组名作形参,以便于实参数组对应,比较直观便于理解。从应用的角度看,用户可以认为有一个形参数组,它从实参数组那里得到起始地址,因此形参数组与实参数组共占同一段内存单元,在调用函数期间,如果改变了形参数组的值,也就是改变了实参数组的值。在主调函数中就可以利用这些已经改变的值。
【例】将数组a中n个整数按相反顺序存放,用一个函数inv来实现交换。实参用数组名a,形参可用数组名,也可用指针变量。
void inv(int x[], int n)//形参x是数组名
{int temp;int m = (n - 1) / 2;for (int i = 0; i <= m; i++){int j = n - 1 - i;//把x[i]和x[j]交换temp = x[i];x[i] = x[j];x[j] = temp;}return;
}
int main()
{int a[10] = { 1,2,3,4,5,6,7,8,9,10 };inv(a, 10);//调用inv函数进行交换for (int i = 0; i < 10; i++){printf("%d ", a[i]);}
}
运行结果:
结果分析:
在main主函数中定义整型数组a,并赋初值。函数inv的形参数组名为x。在定义inv函数时,可以不指定形参数组x的大小(元素个数)。因为形参数组名实际上是一个指针变量,并不是真的开辟一个数组空间(定义实参数组时必须指定数组大小,因为要开辟相应的存储空间)。inv函数的形参n用来接收需要处理的元素的个数。在main函数中有函数调用语句inv(a,10);
,表示要求将a数组的10个元素颠倒排列。如果改为inv(a,5);
,则表示要求将a数组的前5个元素颠倒排列,此时函数inv只处理5个数组元素。函数inv中的m是i值得上限,当i<=m时,循环继续执行;当i>m时,则结束循环过程。
改写代码,将函数inv中得形参x改成指针变量:
void inv(int *x, int n)
{int temp;//定义一个中间变量,用来交换两个变量得值int m = (n - 1) / 2;int* p;//定义一个指针变量pp = x + m;//p指向a[m]元素的地址int* i;//定义一个指针变量ii = x;//i指向数组首元素的地址int* j;//定义一个指针变量jj = x + n - 1;//j指向数组最后一个元素的地址for (i=x;i<=p;i++,j--){//交换i和j所指向的数组元素的值temp = *i;*i = *j;*j = temp;}return;
}
int main()
{int a[10] = { 1,2,3,4,5,6,7,8,9,10 };inv(a, 10);for (int i = 0; i < 10; i++){printf("%d ", a[i]);}
}
改写后,函数inv中得形参有数组名x[]变为了指针变量*x,相应的实参仍是数组名a,即数组a首元素的地址,将它传给形参指针变量x,这时x就指向a[0]。x+m是a[m]元素得地址。设i和j以及p都是指针变量,用它们指向有关元素。交换i和j所指向的数组元素的值,实际上就是交换a[i]和a[j]的值。
运行结果:
5.归纳分析
(1)形参和实参都用数组名
int fun(int x[],int n)
{...
}
int main()
{int a[10];...fun(a,10);...return 0;
}
由于形参数组名x接收了实参数组名首元素a[0]的地址,因此可以认为在函数调用期间,形参数组和实参数组共用一段内存单元,如图所示:
(2)实参用作数组名,形参用指针变量
void fun(int *x,int n)
{...
}
int main()
{int a[10];...fun(a,10);...return 0;
}
实参a为数组名,形参x为int *型的指针变量,调用函数开始后,形参x指向a[0],即x=&a[0],通过x值的改变,可以指向a数组的任一元素,如图所示:
(3)实参形参都用指针变量
void fun(int *x,int n)
{...
}
int main()
{int a[10];int *p=a;...fun(a,10)...return 0;
}
实参p和形参x都是int*型指针变量。先使用实参指针变量p指向数组元素a[0],p的值是&a[0]。然后将p的值传给形参变量x,x的初始值也是&a[0],通过x的值的改变可以使x指向数组a的任一元素,如图所示:
(4)实参为指针变量,形参为数组名
void fun(int x[],int n)
{...
}
int main()
{int a[10];int *p=a;...fun(p,10);...return 0;
}
实参p作为指针变量,它指向a[0]。形参为数组名x,编译系统把x作为指针变量处理,将a[0]的地址传给形参x,使x也指向a[0]。也可以理解为形参数组x和a数组共用同一内存单元,在函数执行过程中可以使x[i]的值发生变化,而x[i]就是a[i]。这样主函数可以使用改变了的数组元素值,如图所示:
【例】指针变量做实参
void inv(int* x, int n)
{int temp;int m = (n - 1) / 2;int* p;p = x + m;int* i;i = x;int* j;j = x + n - 1;for (i = x; i <= p; i++, j--){temp = *i;*i = *j;*j = temp;}return;
}
int main()
{int a[10];int* p = a;//指针变量p指向a[0]//输入数组a的元素for (int i = 0; i < 10; i++, p++){scanf("%d", p);}p = a;//将指针变量p重新指向a[0]inv(p, 10);//调用inv函数,实参是变量pfor (int i = 0; i < 10; i++){printf("%d ", a[i]);}return 0;
}
运行结果:
上面的main函数中的指针变量p是有确定值的。如果在main函数中不设数组,只设指针变量,就会出错。假如把主函数修改如下:
void inv(int* x, int n)
{int temp;int m = (n - 1) / 2;int* p;p = x + m;int* i;i = x;int* j;j = x + n - 1;for (i = x; i <= p; i++, j--){temp = *i;*i = *j;*j = temp;}return;
}
int main()
{int a[10];int* p;//输入数组a的元素for (int i = 0; i < 10; i++){scanf("%d", p+i);}inv(p, 10);//调用inv函数,实参是变量pfor (int i = 0; i < 10; i++){printf("%d ", *(p+i));}return 0;
}
编译时会出错,原因是指针变量p没有确定值,谈不上指向哪个变量:
因此,下面这样的使用是不正确的:
void fun(int x[],int n)
{...
}
int main()
{int *p;...fun(p,10);
}
【注意】如果指针变量做实参,必须先使指针变量有确定的值,指向一个已经定义的对象。
以上4种方法,实质上都是地址的传递。其中(3)和(4)两种只是形式上不同,实际上形参都是使用的指针变量。
【例】使用指针方法对10个整数按由大到小的顺序排列。
【思路】在主函数种定义数组a存放10个整数,定义int*型指针变量p指向a[0]。定义函数sort使数组a种的元素由大到小的顺序排列。在主函数中调用sort函数,用指针变量p作实参。sort函数的形参用数组名。用选择法进行排序。
//定义sort函数,x是形参数组名
void sort(int x[], int n)
{int temp;for (int i = 0; i < n-1; i++){for (int j = i + 1; j < n; j++){if (x[j]>x[i]){temp = x[i];x[i] = x[j];x[j] = temp;}}}
}
int main()
{int a[10];int* p = a;//指针变量p指向a[0]//输入10个整数for (int i = 0; i < 10; i++){scanf("%d", p++);}p = a;//指针变量p重新指向a[0]sort(p, 10);//调用sort函数//输出排序后的10个数组元素for (int i = 0; i < 10; i++){printf("%d ", *p);p++;}return 0;
}
运行结果:
如果sort函数中将x定义为指针变量,在函数中仍可以用x[i]和x[j]这样的形式表示数组元素,它就是x+i和x+j所指的数组元素。
相关文章:

C语言 用数组名作函数参数
当用数组名作函数参数时,如果形参数组中各元素的值发生变化,实参数组元素的值随之变化。 1.数组元素做实参的情况: 如果已经定义一个函数,其原型为 void swap(int x,int y);假设函数的作用是将两个形参(x,y…...
每日一题(980. 不同路径 III)-回溯
题目 980. 不同路径 III 题解思路 表格中值为1的为起始点值为0 的是可以经过的点,但是只能经过一次值为2 的是终点,计算从起点到终点一共有多少种路径 计算出值为0的方格个数,同时找到起点位置当位于终点时候且经过所有的方格为0的点 即为…...
【Python:json常用函数,用于加载和保存json文件】load(), loads(), dump(), dumps()
文章目录 1、load()2、loads()3、dump()4、dumps() json文件为javascript object Notation文件,属于轻量级的数据交换格式,可以用于存储和交换数据。json文件是由类似{ }的key-value映射组成。 1、load() 把json文件加载为Python的数据格式,…...

Flink State 和 Fault Tolerance详解
有状态操作或者操作算子在处理DataStream的元素或者事件的时候需要存储计算的中间状态,这就使得状态在整个Flink的精细化计算中有着非常重要的地位: 记录数据从某一个过去时间点到当前时间的状态信息。以每分钟/小时/天汇总事件时,状态将保留…...

小红书2023“家生活”趋势白皮书
关于报告的所有内容,公众【营销人星球】获取下载查看 核心观点 近年来,年轻人与家的关系愈发紧密。 在小红书上,我们观察到了家居家装内容的蓬勃生长,3 年来相关内容的笔记规模增长了6倍,相关品类的搜索量增加的 3.…...

使用 LangChain 搭建基于 Amazon DynamoDB 的大语言模型应用
LangChain 是一个旨在简化使用大型语言模型创建应用程序的框架。作为语言模型集成框架,在这个应用场景中,LangChain 将与 Amazon DynamoDB 紧密结合,构建一个完整的基于大语言模型的聊天应用。 本次活动,我们特意邀请了亚马逊云科…...
210. 课程表 II Python
文章目录 一、题目描述示例 1示例 2示例 3 二、代码三、解题思路 一、题目描述 现在你总共有 numCourses 门课需要选,记为 0 到 numCourses - 1。给你一个数组 prerequisites ,其中 prerequisites[i] [ai, bi] ,表示在选修课程 ai 前 必须 …...
【LeetCode 算法】Linked List Cycle II 环形链表 II
文章目录 Linked List Cycle II 环形链表 II问题描述:分析代码哈希快慢指针 Tag Linked List Cycle II 环形链表 II 问题描述: 给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。 如果链…...

蒸散发与植被总初级生产力估算
目标 熟悉蒸散发ET及其组分(植被蒸腾Ec、土壤蒸发Es、冠层截留Ei)、植被总初级生产力GPP的概念和碳水耦合的基本原理;掌握利用Python与ArcGIS工具进行课程相关的操作;熟练掌握国际上流行的Penman-Monteith模型,并能够…...

uniapp微信小程序底部弹窗自定义组件
基础弹窗效果组件 <template><view><viewclass"tui-actionsheet-class tui-actionsheet":class"[show ? tui-actionsheet-show : ]"><view class"regional-selection">底部弹窗</view></view><!-- 遮罩…...

人工智能的最新进展:2024年将会发生什么?
文章目录 2024年AI最新发展2024年AI具体应用2024年AI的具体预测 ✍创作者:全栈弄潮儿 🏡 个人主页: 全栈弄潮儿的个人主页 🏙️ 个人社区,欢迎你的加入:全栈弄潮儿的个人社区 📙 专栏地址&#…...

使用Golang实现一套流程可配置,适用于广告、推荐系统的业务性框架——组合应用
在《使用Golang实现一套流程可配置,适用于广告、推荐系统的业务性框架——简单应用》中,我们看到了各种组合Handler的组件,如HandlerGroup和Layer。这些组件下面的子模块又是不同组件,比如LayerCenter的子组件是Layer。如果此时我…...
DNS入门学习:DNS缓存的原理和作用(中科三方)
在实际业务场景中,DNS解析过程并不总是严格遵循从根域名服务器、顶级域名服务器再到权威域名服务器的一级级查询过程,这只是一个标准状态。为了节省全球查询的时间,同时减轻各级服务器的解析压力,DNS系统中引入了缓存机制。本文中…...

Linux虚拟机安装tomcat(图文详解)
目录 第一章、xshell工具和xftp的使用1.1)xshell下载与安装1.2)xshell连接1.3)xftp下载安装和连接 第二章、安装tomcat1.1)关闭防火墙,传输tomcat压缩包到Linux虚拟机12)启动tomcat 第一章、xshell工具和xf…...

Matlab对TMS320F28335编程--SVPWM配置互补PWM输出
前言 F28335中断 目的:FOC的核心算法及SVPWM输出,SVPWM的载波频率10kHz,SVPWM的每个周期都会触发ADC中断采集相电流,SVPWM为芯片ePWM4、5、6通道,配置死区 1、配置中断SVPWM进ADC中断,查上表知CPU1,PIE1 …...

MySQL数据库——多表操作
文章目录 前言多表关系一对一关系一对多/多对一关系多对多关系 外键约束创建外键约束插入数据删除带有外键约束的表的数据删除外键约束 多表联合查询数据准备交叉连接查询内连接查询外连接查询左外连接查询右外连接查询满外连接查询 子查询子查询关键字ALL 关键字ANY 和 SOME 关…...

Java版本spring cloud + spring boot企业电子招投标系统源代码 tbms
功能模块: 待办消息,招标公告,中标公告,信息发布 描述: 全过程数字化采购管理,打造从供应商管理到采购招投标、采购合同、采购执行的全过程数字化管理。通供应商门户具备内外协同的能力,为…...

css实现,正常情况下div从左到右一次排列,宽度超出时,右侧最后一个div固定住,左侧其他div滚动
需求:正常情况下 宽度超出时: 实现: <templete><div class"jieduanbox"><div v-for"(item, index) in stageList" :key"index" style"display: inline-block">.......</div><div class"rightBtn&q…...
【Linux手动搭建Sftp,创建用户、用户组及删除用户】
SFTP (Secure File Transfer Protocol)是一种安全的文件传输协议,基于SSH协议进行加密传输。在进行文件传输时,SFTP客户端通过SSH协议与服务器进行连接,并且通过使用公钥和/或密码进行身份验证,从而确保传输…...

云上 Index:看「简墨」如何为云原生打造全新索引
拓数派首款数据计算引擎 PieCloudDB Database 是一款全新的云原生虚拟数仓。为了提升用户使用体验,提高查询效率,在实现存算分离的同时,PieCloudDB 设计与打造了全新的存储引擎「简墨」等模块,并针对云场景和分析型场景设计了高效…...

linux之kylin系统nginx的安装
一、nginx的作用 1.可做高性能的web服务器 直接处理静态资源(HTML/CSS/图片等),响应速度远超传统服务器类似apache支持高并发连接 2.反向代理服务器 隐藏后端服务器IP地址,提高安全性 3.负载均衡服务器 支持多种策略分发流量…...

51c自动驾驶~合集58
我自己的原文哦~ https://blog.51cto.com/whaosoft/13967107 #CCA-Attention 全局池化局部保留,CCA-Attention为LLM长文本建模带来突破性进展 琶洲实验室、华南理工大学联合推出关键上下文感知注意力机制(CCA-Attention),…...

渗透实战PortSwigger靶场-XSS Lab 14:大多数标签和属性被阻止
<script>标签被拦截 我们需要把全部可用的 tag 和 event 进行暴力破解 XSS cheat sheet: https://portswigger.net/web-security/cross-site-scripting/cheat-sheet 通过爆破发现body可以用 再把全部 events 放进去爆破 这些 event 全部可用 <body onres…...

MySQL 8.0 OCP 英文题库解析(十三)
Oracle 为庆祝 MySQL 30 周年,截止到 2025.07.31 之前。所有人均可以免费考取原价245美元的MySQL OCP 认证。 从今天开始,将英文题库免费公布出来,并进行解析,帮助大家在一个月之内轻松通过OCP认证。 本期公布试题111~120 试题1…...
MySQL中【正则表达式】用法
MySQL 中正则表达式通过 REGEXP 或 RLIKE 操作符实现(两者等价),用于在 WHERE 子句中进行复杂的字符串模式匹配。以下是核心用法和示例: 一、基础语法 SELECT column_name FROM table_name WHERE column_name REGEXP pattern; …...
Android Bitmap治理全解析:从加载优化到泄漏防控的全生命周期管理
引言 Bitmap(位图)是Android应用内存占用的“头号杀手”。一张1080P(1920x1080)的图片以ARGB_8888格式加载时,内存占用高达8MB(192010804字节)。据统计,超过60%的应用OOM崩溃与Bitm…...
Spring Security 认证流程——补充
一、认证流程概述 Spring Security 的认证流程基于 过滤器链(Filter Chain),核心组件包括 UsernamePasswordAuthenticationFilter、AuthenticationManager、UserDetailsService 等。整个流程可分为以下步骤: 用户提交登录请求拦…...

前端开发者常用网站
Can I use网站:一个查询网页技术兼容性的网站 一个查询网页技术兼容性的网站Can I use:Can I use... Support tables for HTML5, CSS3, etc (查询浏览器对HTML5的支持情况) 权威网站:MDN JavaScript权威网站:JavaScript | MDN...

AxureRP-Pro-Beta-Setup_114413.exe (6.0.0.2887)
Name:3ddown Serial:FiCGEezgdGoYILo8U/2MFyCWj0jZoJc/sziRRj2/ENvtEq7w1RH97k5MWctqVHA 注册用户名:Axure 序列号:8t3Yk/zu4cX601/seX6wBZgYRVj/lkC2PICCdO4sFKCCLx8mcCnccoylVb40lP...
32位寻址与64位寻址
32位寻址与64位寻址 32位寻址是什么? 32位寻址是指计算机的CPU、内存或总线系统使用32位二进制数来标识和访问内存中的存储单元(地址),其核心含义与能力如下: 1. 核心定义 地址位宽:CPU或内存控制器用32位…...