C语言数组当参数传递

C语言数组当参数传递在学习C语言的过程中遇到数组作为参数传递的问题一维数组:#includeinttest2(inta[]){ for(inti=0;i<5;i++){ printf("%d",a[i]); }}intmain(){ inta[5]={1,2,3,4,5},*p; p=a; test2(a); }这样我们可以很顺利的在test去遍历这个

大家好,又见面了,我是你们的朋友全栈君。

在学习C语言的过程中遇到数组作为参数传递的问题

一维数组:

#include <stdio.h>

int test2(int a[]){
	for(int i=0;i<5;i++){
		printf("%d",a[i]);
	}
}
int main(){
	int a[5] = {1,2,3,4,5},*p;
	p = a;
	test2(a);
	
}

这样我们可以很顺利的在test去遍历这个数组a,当然我们还可能传递指针:

int test1(int *p){
	for(int i=0;i<5;i++){
		printf("%d",p[i]);//我们在这里还可以用)*(p+i)来输出数组中的值
	}	
}

int main(){
	int a[5] = {1,2,3,4,5},*p;
	p = a;
	test1(p);
	
}

一般来数参数的传递是值传递,也就是说实参传给形参,形参发生改变时实参并不会改变,(单向)但是数组在传递的时候是地址传递,只要形参发生了变化,实参也会发生变化(双向)。

这样传递数组就会发现一个问题,我没有办法获取到数组的长度。获取数组的长度我们一般用:

sizeof(a)/sizeof(int)

int test2(int a[]){
	int n = sizeof(a)/sizeof(int);
	for(int i=0;i<n;i++){
		printf("%d ",a[i]);
		a[i]++;
	}
}

我们会发现n的值一直是2!为什么会这样呢!?


因为,a是函
数参数,到了本函数中,a只是一个指针(地址,系统在本函数运行时,是不知道a所表示的地址有多大的数据存储
空间,这里只是告诉函数:一个数据
空间首地址
),所以,sizoef(a)的结果是指针变量a占内存的大小
,一般在64位机上是8个字节。
int类型是4个字节,所以,结果永远是2,
因此
,我们要向获取数组长度要怎么办呢?

我可以在初始化数组的地方获取到数组的长度,作为参数传递过来:

int test2(int a[],int n){
	for(int i=0;i<n;i++){
		printf("%d ",a[i]);
		a[i]++;
	}
}

int main(){
	int a[5] = {1,2,3,4,5},*p;
	int n = sizeof(a)/sizeof(int);
	test2(a,n);

	
}

这样做我们可以获取到数组的长度。

二维数组:

二维数组作为参数传递是后我们不可以像以为数组那样直接,如:

void test1(int a[][]){
	
	for(i = 0; i < 5; i++){
		for(j = 0; j < 5; j++){
			printf("%d ",a[i][j]);
		}
	}
}

int main(){
	int a[5][5],i,j;
	for(i = 0; i < 5; i++){
		for(j = 0; j < 5; j++){
			a[i][j] = i*5 + (j +1);
		}
	}
	test1(a);
	return 0;	
}

会发现编译都编译不通过,报“[Error] declaration of ‘a’ as multidimensional array must have bounds for all dimensions except the first”这个错,意思是多维数组的定义必须有一个除第一个之外的所有维度的边界,比如:

void test1(int a[][5]){
	
	for(int i = 0; i < 5; i++){
		for(int j = 0; j < 5; j++){
			printf("%d ",a[i][j]);
		}
	}
}

这样就OK了,但是我们是动态分配的数组不知道这个维度是多少的时候怎么办?这时候我们可以用指针当做一维数组来操作:

void test1(int *p,int n){
	for(int i = 0; i < n; i++){
		printf("%d ",p[j]);
	}
}


int main(){
	int a[5][5],i,j;
	int *p;
	p = &a[0][0];
	for(i = 0; i < 5; i++){
		for(j = 0; j < 5; j++){
			a[i][j] = i*5 + (j +1);
		}
	}
	test1(p,25);
	return 0;	
}

这样我们发现不能更灵活的去定位到某一行某一列,这样我们需要手工改变寻址方式:

void test2(int  m,int  n,int **p){//m,n是行和列,
	for(int i = 0; i < m; i++){
		for(int j = 0; j < n; j++){
			printf("%d ",*((int *)p+n*i+j));
		}
	}
}

int main(){
	int a[5][5],i,j;
	
	for(i = 0; i < 5; i++){
		for(j = 0; j < 5; j++){
			a[i][j] = i*5 + (j +1);
		}
	}
	test2(5,5,(int **)a);
	return 0;	
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/157305.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 物联网用什么系统(物联网技术)

    前言  操作系统是物联网时代的战略制高点,今天PC和手机时代的操作系统霸主未必能在物联网时代延续霸业。操作系统产业的规律是,当垄断已经形成,后来者就很难颠覆,只有等待下一次产业浪潮。如今,一个全新的、充满想象空间的操作系统市场机会正在开启。  如此关键的产业环节必然是兵家必争之地。ARM、谷歌、微软、华为、阿里、海尔等国内外著名的IT企业纷纷推出物联网操作系统,整个产业呈现出群雄逐鹿的壮

    2022年4月13日
    219
  • OCX控件签名

    OCX控件签名即便是经常被人批评,但是OCX,还是存在很多年了,在经后很多年,它都还将存在着。因为在有的环境下,简单的b/s开发技术是达不到需求的。比如访问硬件设备,比如安全防护。因此我们还要用到OCX的。但是要想在正式的场合下,在WEB界面上用OCX,需要进行签名。否则很多时候浏览器会禁止加

    2022年7月13日
    26
  • 几种常见mybatis分页实现[通俗易懂]

    几种常见mybatis分页实现[通俗易懂]mybatis框架分页实现,有几种方式,最简单的就是利用原生的sql关键字limit来实现,还有一种就是利用interceptor来拼接sql,实现和limit一样的功能,再一个就是利用PageHelper来实现。这里讲解这三种常见的实现方式:无论哪种实现方式,我们返回的结果,不能再使用List了,需要一个自定义对象Pager。packagecom.xxx.mybatis.bean;…

    2022年10月20日
    4
  • navicat 激活码 mac【2021最新】

    (navicat 激活码 mac)好多小伙伴总是说激活码老是失效,太麻烦,关注/收藏全栈君太难教程,2021永久激活的方法等着你。IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/100143.html1STL5S9V8F-eyJsaWNlbnNlSWQi…

    2022年3月27日
    95
  • libzplay库

    libzplay目前,非开源,只可以在windows上应用;关于MP3文件播放:通常步骤是:获取MP3相关参数->解码->相关平台播放音频接口播放声音;可以播放解码播放MP3的库很多,如果VLC,ffplay,或者directshow,解码库一般可以用lame,播放播放库可以用SDL,或者Windows上的waveout,directsound等很多方法,这里例举

    2022年4月9日
    45
  • css学习_css补充知识「建议收藏」

    css学习_css补充知识「建议收藏」css学习_css补充知识

    2022年4月22日
    49

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号