文章目录

  • 1 引例
  • 2 观点1 这种使用方法是错误的
  • 3 观点2 根本不需要这么做
  • 4 二维数组做函数参数的方法
    • 4.1 方法1
    • 4.2 方法2
    • 4.3 方法3
  • 5 与Java的不同

1 引例

下面的程序很简单,定义了一个PrintMatrix函数将一个二维数组以矩阵的形式打印出来。

#include <stdio.h>
#define SIZE (4)void PrintMatrix(int **arr)
{for (int i = 0; i < SIZE; i++) {for (int j = 0; j < SIZE; j++) {printf(" %d", arr[i][j]);}printf("\n");}
}int main(void)
{int matrix[SIZE][SIZE] = {{1, 2, 3, 4},{5, 6, 7, 8},{1, 2, 3, 4},{5, 6, 7, 8}};PrintMatrix(matrix);return 0;
}

当我们编译这个程序的时候会出现下面的错误信息:

error: cannot convert 'int (* )[4]' to 'int**'

按理来说,一维数组对应着一级指针,就像我们经常使用的,作为函数参数的时候也不理外:

int arr[4] = {1, 2, 3, 4};
int *p = arr;

那么二维数组对应着二级指针怎么就出错了呢?

要想解释清楚这个问题,首先需要知道二级指针和二维数组的定义:

  • 二维数组:二维数组本质上是以数组作为数组元素的数组,即“数组的数组”;
  • 二级指针:指向指针的指针。

下面将会从两种观点来分析这个问题。

2 观点1 这种使用方法是错误的

举一个简单的例子,定义一个二维数组。

int a[2][2] = {1, 2, 3, 4};

假如用一个二级指针指向它。

int **p = a;

运行如下代码(打印了a的值和aa[0]a[0][0] 的地址):

    printf("%#x \n", a);printf("%#x \n", &a);printf("%#x \n", &a[0]);printf("%#x \n", &a[0][0]);

运行的结果是输出的四行结果是完全一样的(不同计算机输出的地址可能不同):

0x9ffe40
0x9ffe40
0x9ffe40
0x9ffe40

既然 int **p = a;,可以推出 p 保存的是 a[0][0] 的地址,那么用 * 号可以对其解引用指针而取出地址的内容,于是测试以下代码:

    printf("%d \n", *&a[0][0]);

p 指针进行了一次解引用,和预期的一样,输出了二维数组 a 的第一个元素 1,理所当然的 p 是二级指针,那么也可以对其进行二次解引用,当我们对其进行二次解引用的时候,编译器报错了:

error: invalid type argument of unary '*' (have 'int')

还有一点值得注意的是,在二维数组中,a + 1,可以移动一个元素的位置(也就是移动一个一维数组的大小),而用二级指针,p + 1 永远移动一个 sizeof(int) 的大小(本例中)。

3 观点2 根本不需要这么做

学过 C/C++ 的朋友应该都知道,一位数组在内存中是线性排列的,那么二维数组也不理外,它也是线性排列的。

假如只知道 arr 的地址,那么能得到 arr[1][1] 的地址吗?
答案是可以的,因为数组在内存中是线性排列,那么就代表着只要知道数组的起始地址,就可以通过一次取址获得任意元素的位置,所以根本不需要二次取址。

这也就解释了为什么声明二维数组的时候,编译器根本“不关心”一维的大小,以至于可以省略掉它,如:

int a[][2];

4 二维数组做函数参数的方法

4.1 方法1

void fun(int arr[2][2]);

这种方法导致只能处理2行2列的int型数组,即固定了数组大小。

4.2 方法2

void fun(int arr[][2]);

可以省略一维的大小,这种方法的限制略微宽松了一些,但是还是只能处理每行是2个整数长度的数组。

4.3 方法3

void fun(int (*arr)[2]);

这个方法需要重点讨论,这里引入了一个新的概念叫做数组指针。

int (*arr)[2];

在解释这个概念之前要比较一下数组指针和指针数组的不同,以防止混淆:

  • 指针数组(array of pointers):即用于存储指针的数组,也就是数组元素都是指针,如 int* a[4],表示数组 a 中的元素都为 int 型指针;
  • 数组指针(a pointer to an array):即指向数组的指针,如 int (*a)[4],表示指向数组 a 的指针。

通过上述概念就知道了应该用数组指针指向一个二维数组,以下是正确的示例:

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

5 与Java的不同

Java中声明一个二维数组:

int[][] arr = new int[2][2];

不同于 C/C++ 中的:

int arr[2][2];

也不同与 C/C++ 中的:

int (*p)[2] = new int[2][2];

在 Java 中则是分配了一个包含 2 个指针的数组,指针数组的每个元素包含一个一维数组,在 C++ 中的声明如下:

int **p = new int *[2];

C语言-二维数组做函数的参数相关推荐

  1. C语言 二维数组做函数参数的几种情况

    (1)       实参为数组元素地址,虚参为元素类型指针 (a)       调用函数:fun(*a,3*4);  //*a也可为a[0] or &a[0][0] 函数: fun(int * ...

  2. C++二维数组做函数参数

    C++二维数组做函数参数 二维数组做函数参数的形式主要有: /对于一个m行n列int元素的二维数组 //函数f的形参形式 f(int daytab[m][n]) {...}//以下两种可以忽略行数 f ...

  3. C语言 | 二维数组作为函数参数

    1024G 嵌入式资源大放送!包括但不限于C/C++.单片机.Linux等.关注微信公众号[嵌入式大杂烩],回复1024,即可免费获取! 偶然间发现C语言二维数组作为函数的参数是个比较容易出错的问题. ...

  4. C语言二维数组作为函数参数传递

    二维数组存放方式 二维数组在内存中是按行存放的,先存储第一行,在接着存储第二行-.. 二维数组作为函数参数 二维数组作为函数的参数,实参可以直接使用二维数组名,在被调用函数中可以定义形参所有维数的大小 ...

  5. 二维数组作为函数的参数和返回值

    1. 二维数组作为函数的参数  (1)错误的做法:将二维数组传递给二维指针 //实参是二维数组,形参是int**,这样做的话编译器会报错 int main() {int arr[3][4] = { 1 ...

  6. C++笔记 二维数组作为函数的参数详解 三种传参的方法总结 注意要点总结

    文章目录 1.C/C++ 二维数组作为函数的参数 2.不合法写法总结 3.测试案例 3.1 传参方式1 3.2 传参方式2 3.3 传参方式3 4.运行结果 5.总结 1.C/C++ 二维数组作为函数 ...

  7. C语言多维数组做函数参数技术推演

    多维数组做函数参数技术 C语言中只会以机械式的值拷贝的方式传递参数(实参把值传给形参) 二维数组参数同样存在退化的问题 等价关系 C语言中只会以机械式的值拷贝的方式传递参数(实参把值传给形参) int ...

  8. C语言 二维数组作为函数参数的4种方式

    前言 多维数组中,二维数组是最常用的一种.在C语言编程中,二维数组的定义.取值以及赋值都比较容易,与一维数组类似.然而,在将二维数组作为函数参数传递时,参数结构较复杂,难以理解.本文章是实用型文章,注 ...

  9. C语言多维数组做函数参数退化原因大剖析

    多维数组做函数参数退化原因 多维数组做函数参数退化原因大剖析 多维数组做函数参数退化原因大剖析 //证明一下多维数组的线性存储 //线性打印 void printfArray411(int *arra ...

最新文章

  1. ios iphonex适配
  2. 《Microsoft Sql server 2008 Internals》读书笔记--第十一章DBCC Internals(11)
  3. 这个神奇的库,可以将数据平滑化并找到异常点
  4. 设计一个具有等待队列的连接池
  5. 解决 Cannot open D:\Program Files\Anaconda3\Scripts\pip-script.py 问题
  6. linux 操作mysql 数据库命令_Linux 操作数据库命令
  7. 全排列的递归与非递归形式
  8. Android 屏幕适配攻略(二)单位dp与px来表示控件的尺寸
  9. 数据结构:分块-区间加法、区间乘法和单点查询
  10. 【251】◀▶IEW-Unit16
  11. 标准物模型:设备无缝对接,IOT界的福音
  12. out memory 内存溢出总结
  13. VISUAL STUDIO 与 MATLAB实现混合编程
  14. ubuntu16.04安装google拼音输入法
  15. poi excel 添加水印
  16. linux+硬盘rd5,BackTrack5(BT5)硬盘安装完美教程 亲测可用
  17. 部分商誉确认法和全部商誉确认法
  18. 论文阅读:Axiomatic Characterization of Data-Driven Influence Measures for Classification
  19. arcgis用python字段自动编号_属性表字段自动编号
  20. 企业邮箱哪个最好用?哪个企业邮箱更优惠?费用是多少?

热门文章

  1. Navicat远程连接不上mysql解决方案
  2. python模块介绍-locustio:性能测试工具locustio
  3. Android之Handler有感(二)
  4. 【python技巧】“”、“”等符号操作
  5. Caffe---Pycaffe进行网络结构(xxx.prototxt)可视化
  6. 支付宝接口参数调用相关
  7. 动态可订制属性的 PropertyGrid(转载)
  8. Django中载入js和css文件
  9. 探秘腾讯Android手机游戏平台之不安装游戏APK直接启动法
  10. JXOI2018做题笔记