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)
上一篇 2022年7月11日 上午10:16
下一篇 2022年7月11日 上午10:16


相关推荐

  • MySQL 5.7的原生JSON数据类型使用

    MySQL 5.7的原生JSON数据类型使用

    2022年2月18日
    49
  • [Python人工智能] 四.神经网络和深度学习入门知识

    [Python人工智能] 四.神经网络和深度学习入门知识从本篇文章开始 作者正式开始研究 Python 深度学习 神经网络及人工智能相关知识 前两篇文章讲解了神经网络基础概念 Theano 库的安装过程及基础用法 theano 实现回归神经网络 这篇文章主要讲解机器学习的基础知识 再通过 theano 实现分类神经网络 主要是学习 莫烦大神 网易云视频的在线笔记 后面随着深入会讲解具体的项目及应用 基础性文章 希望对您有所帮助 也建议大家一步步跟着学习 同时文章

    2026年3月18日
    2
  • 商标注册_企业软件

    商标注册_企业软件开发软件时,当用到商业用途时,注册码与激活码就显得很重要了。现在的软件激活成功教程技术实在在强了,各种国内外大型软件都有注册机制,但同时也不断地被激活成功教程。下面发的只是一个常用版本,发出源码被破就更容易了,但我们学习的是技术。当然也为以后自己的软件不会被轻易激活成功教程。第一步。根据卷标,CPU序列号,生成机器码//取得设备硬盘的卷标号       publicstaticstringG

    2026年4月16日
    5
  • nanomsg使用_jmeter下载安装教程

    nanomsg使用_jmeter下载安装教程最近在构建一个中间层的通信架构,本来想用dbus,在实验过程中发现dbus对于国产系统支持版本比较低,安装比较麻烦,今天无意中看中了nanomsg,尽管没有dbus那么强悍的生态,但基本能满足需求。

    2022年8月4日
    11
  • 布隆过滤器的原理_板框过滤器

    布隆过滤器的原理_板框过滤器引言布隆过滤器被广泛用于

    2022年10月7日
    3
  • 进程间的7种通信方式_linux 进程间通信

    进程间的7种通信方式_linux 进程间通信1无名管道通信无名管道(pipe):管道是一种半双工的通信方式,数据只能单向流动,而且只能在具有亲缘关系的进程间使用。进程的亲缘关系通常是指父子进程关系。2高级管道通信高级管道(popen):将另一个程序当做一个新的进程在当前程序进程中启动,则它算是当前程序的子进程,这种方式我们成为高级管道方式。3有名管道通信有名管道(namedpipe):有名管道也是半双工的通信方式,但是它允许

    2022年10月11日
    4

发表回复

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

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