c 向函数传递多维数组

 

注:本文完全转载,
 

c中对多维数组的支持并不是特别的完美,导致了许多的操作对于新手来说晦涩难懂。当然如果只是从语法本身出发考虑是很难理解为什么需要这么做的,但是从编译器的角度出发恰好可以获取意想不到的效果。这起决于编译也需要一定的条件才能判断,而这个条件往往就是为什么这样子可以的原因。

向函数传递数组,相信会记起一条规则“数组名会被改写成一个指针参数”。但是这个规则并不是递归定义的。也就是说在一维数组时,数组名确实被改写成了指针参数。但是数组的数组则会被改写成“数组的指针”,而不是“指针的指针”。

以下是一段示例程序,将帮助你理解:

#include "stdio.h"

int g_ia[10][20];

char *g_p_ca[]={"abcd","efg","hijk"};

//session 1:形参直接定义为 int l_ia[10][20]
//迫使函数只处理10行20列的int型数组
void array_func_exact_multiple_param(int l_ia[10][20]){
printf("l_ia[10][20]/n");
return ;
}

//session 2:形参定义为 int l_ia[][20]
//多维数组最主要的一维的长度(最左边一维)不必显示说明
//所有的函数都必须知道数组其他维的确切长度和数组的基地址。
//数组其他维的确切长度是为编译器提供一个确切的步长,以便计算地址
void array_func_exact_less_param(int l_ia[][20]){
printf("l_ia[][20]/n");
return ;
}

//session 3:形参定义为 int (*p_ia)[20]
//同样p_ia 是指针,指向一个长度为20的int型数组
//编译将能获知每次移动的步长以及基地址
void array_point_func(int (*p_ia)[20]){
printf("int (*p_ia)[20]/n");
return ;
}


//session 4:形参定义为 int **p_p_ia
//如果直接将 int g_ia[10][20] 作为实参
//掉用之后会发现,编译器将在函数内部无法获知数组的该移动的步长
//也就是说这种定义方式仅仅提供给了编译器基地址,
//p_p_ia[i][j]的地址为 基地址 + i*(p_p_ia[]的步长) + j*(p_p_ia[][]的步长)
//明显现在已经无法预知 p_p_ia[] 的步长,
//仅仅知道 p_p_ia[][] 的步长为 int,4个字节
//所以在调用时会发现编译报出错误 cannot convert parameter 1 from 'int [10][20]' to 'int ** '
void int_point_point_func(int **p_p_ia){
printf("int **p_p_ia/n");
return ;
}

//session 5: 形参定义为char **p_p_ca
//实参传递进来的是char *g_p_ca[]
//明显当使用的是指针数组时,移动的步长是明确的,就是指针的长度
//而指针数组的地址就是基地址,所以当使用这种情况时,编译是通过的
//但是得注意,因为使用指针数组时,因为分配的空间不一定等长
//即有可能是锯齿状数组,所以在使用时只能是固定长度或者有默认的结束符
//但是固定长度在锯齿状下是相对行不通的
//所以只能选择默认的结束符,一般使用这种情况的就是字符串数组
//因为有默认的结束符'/0'
void char_point_point_func(char **p_p_ca){
printf("char **p_p_ca/n");
return ;
}

int main(int argc, char* argv[])
{
printf("Hello World!/n");

//session 6:编译通过
array_func_exact_multiple_param(g_ia);
array_func_exact_less_param(g_ia);
array_point_func(g_ia);

//session 7:编译无法通过,因为无法提供给编译确切的移动步长
//导致编译器只知道基地址,但是没办法计算移动的距离
//也无法计算出最终g_ia[i][j] 的地址
//int_point_point_func(g_ia);

//session 8:编译通过
//这种情况是 session 7的特殊情况
//因为实参使用的是指针数组,移动的步长是固定的,因而
//编译是可以接受的
char_point_point_func(g_p_ca);


//conclusion :总而言之,在多维数组传递,需要提供给编译器确切的步长和基地址
//这是编译器能够正确识别数组,计算出最终地址的必要条件
//
//多维数组和数组在向函数传递的时候,进行的转换是不一样的
//“数组名被改写成一个指针参数”规则并不是递归定义的。
//数组的数组会被改写成“数组的指针”,而不是“指针的指针”
//    实参   所匹配的形参
//一维数组  char a[10]   char *a      指针
//数组的数组 char c[8][10]  char (*c)[10]//(行指针) 数组指针
//指针数组  char *c[15]   char **c     指针的指针
//数组指针  char (*c)[64]  char (*c)[64]    不改变
//指针的指针 char **c   char **c     不改变

return 0;
}

### C语言多维数组作为函数参数的定义与使用 在C语言中,多维数组可以作为一种重要的数据结构被传递函数中进行操作。以下是关于如何定义和使用多维数组作为函数参数的具体方法及其示例代码。 #### 定义多维数组作为函数参数 当我们将一个多维数组作为参数传递函数时,需要注意的是,在C语言中,多维数组本质上是以一维方式存储的连续内存块。因此,为了正确访问多维数组的内容,必须指定其维度信息[^1]。对于二维数组而言,通常只需要显式声明列数即可,而行数可以在运行时动态传入。 下面展示了一个简单的例子来说明这一过程: ```c #include <stdio.h> // 声明一个接受二维数组作为参数的函数原型 void printArray(int rows, int cols, int array[][cols]) { for (int i = 0; i < rows; ++i) { for (int j = 0; j < cols; ++j) { printf("%d ", array[i][j]); } printf("\n"); } } int main() { // 初始化一个二维数组 int myArray[3][4] = {{1, 2, 3, 4}, {5, 6, 7, 8}, {9, 10, 11, 12}}; // 调用printArray函数并打印数组内容 printArray(3, 4, myArray); return 0; } ``` 在这个例子中,`printArray` 函数接收三个参数:行数 `rows`、列数 `cols` 和二维数组本身 `array[][]`。注意这里我们通过 `[cols]` 明确指定了每行有多少列。 #### 使用指针形式表示多维数组 除了上述直接的方式外,还可以利用指针的形式来表达多维数组。这种方式更加灵活,尤其适合处理大小不固定的数组情况[^2]。例如: ```c #include <stdio.h> #include <stdlib.h> // 使用指针形式的参数 void printArrayPointerStyle(int **arr, int rows, int cols){ for(int i=0;i<rows;i++){ for(int j=0;j<cols;j++) { printf("%d ", arr[i][j]); } printf("\n"); } } int main(){ const int ROWS = 3; const int COLS = 4; // 动态分配二维数组空间 int* matrix[ROWS]; for(int i=0;i<ROWS;i++) matrix[i]=(int*)malloc(COLS*sizeof(int)); // 给矩阵赋值 int value = 1; for(int i=0;i<ROWS;i++) for(int j=0;j<COLS;j++) matrix[i][j]=value++; // 输出矩阵 printArrayPointerStyle(matrix, ROWS, COLS); // 清理内存 for(int i=0;i<ROWS;i++) free(matrix[i]); return 0; } ``` 此程序展示了如何创建一个由指针组成的数组,并将其当作二维数组对待。这种方法的好处在于能够更方便地管理动态尺寸的数组。 #### 关于函数指针的应用扩展 如果涉及到更为复杂的场景,比如需要根据不同条件执行不同的计算逻辑,则可能需要用到函数指针的概念[^3]。这不仅限于单个数值的操作,也可以应用于整个数组甚至更高纬度的数据集上。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值