当前位置: 首页 > ai >正文

【C语言干货】二维数组传参本质

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录

  • 前言
  • 一、二维数组的内存布局
    • 1.二维数组的实质
    • 2.二维数组的地址关系
  • 二、二维数组传参的本质
    • 1.参数传递的退化机制
    • 2.三种等效的函数声明方式
  • 总结

前言


提示:以下是本篇文章正文内容,下面案例可供参考

一、二维数组的内存布局

1.1 二维数组的实质

二维数组本质上是一个"数组的数组",即每个元素本身又是一个数组。例如:

int arr[3][4] = {{1, 2, 3, 4},{5, 6, 7, 8},{9, 10, 11, 12}
};

在内存中,二维数组是按行优先顺序连续存储的,实际内存布局为:
1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12

1.2 二维数组的地址关系

对于上述arr[3][4]

  • arr是二维数组名,类型是int (*)[4](指向包含4个int的数组的指针)

  • arr[0]是第一行一维数组的数组名,类型是int *

  • &arr[0][0]是第一个元素的地址,类型是int *

printf("arr: %p\n", (void*)arr);
printf("arr+1: %p\n", (void*)(arr+1)); // 跳过一行(16字节)
printf("arr[0]: %p\n", (void*)arr[0]);
printf("arr[0]+1: %p\n", (void*)(arr[0]+1)); // 跳过一个元素(4字节)
printf("&arr[0][0]: %p\n", (void*)&arr[0][0]);

二、二维数组传参的本质

2.1 参数传递的退化机制

当二维数组作为函数参数传递时,会发生"数组到指针"的退化(decay):

  1. 第一维会退化为指针

  2. 第二维必须明确指定大小

  3. 传递的实际上是第一行的地址

2.2 三种等效的函数声明方式

方式一:完整二维数组形式
void func(int arr[3][4], int rows, int cols);
方式二:省略第一维大小
void func(int arr[][4], int rows, int cols);
方式三:数组指针形式
void func(int (*arr)[4], int rows, int cols);

这三种声明在编译器看来完全等价,都会被视为int (*)[4]类型的参数。


总结

1 核心要点总结

  1. 二维数组传参本质是传递指向第一行的指针

  2. 必须指定第二维的大小,以便编译器计算行偏移

  3. 三种声明方式完全等价,推荐使用int (*arr)[N]形式以明确指针本质

  4. 动态分配的"二维数组"需要不同的传参方式

  5. C99变长数组提供了更灵活的解决方案

2 最佳实践建议

  1. 对于固定大小的二维数组,使用数组指针形式传参:

    void func(int (*arr)[4], int rows);
  2. 对于动态分配的二维数组,使用二级指针传参:

    void func(int **arr, int rows, int cols);
  3. 在C99及以上环境中,考虑使用变长数组语法提高灵活性

  4. 始终传递行数和列数作为额外参数,避免在函数内部尝试获取数组大小

  5. 如果函数不修改数组内容,添加const修饰符:

    void func(const int (*arr)[4], int rows);

理解二维数组传参的本质,关键在于认识其内存布局和"数组到指针"的退化机制。掌握了这些原理,你就能游刃有余地处理各种二维数组相关的函数设计了。

http://www.xdnf.cn/news/5347.html

相关文章:

  • C++23 views::repeat (P2474R2) 写一篇博客
  • Flutter - UIKit开发相关指南 - 导航
  • 深入理解 Java 适配器模式:架构设计中的接口转换艺术
  • 集成灶十大品牌对比
  • Nodejs核心机制
  • 说说Redis的内存淘汰策略?
  • 超市销售管理系统 - 需求分析阶段报告
  • Fiori学习专题四十:单一控件
  • 汇编学习——iOS开发对arm64汇编的初步了解
  • Spring Boot项目(Vue3+ElementPlus+Axios+MyBatisPlus+Spring Boot前后端分离)
  • 微服务架构实战:从服务拆分到RestTemplate远程调用
  • DINOv2
  • Spring框架(一)
  • Spring AI(3)——Chat Memory
  • skopeo工具详解
  • 成功案例:塔能精准节能技术为核心的工厂节能
  • GitHub打开缓慢甚至失败的解决办法
  • RTOS优先级翻转
  • 论文解读:MP-SfM: Monocular Surface Priors for Robust Structure-from-Motion
  • 22.第二阶段x64游戏实战-分析周围对象类型
  • SHAP分析!Transformer-BiLSTM组合模型SHAP分析,模型可解释不在发愁!
  • 分享一个可以用GPT打标的傻瓜式SD图片打标工具——辣椒炒肉图片打标助手
  • 04.three官方示例+编辑器+AI快速学习webgl_animation_skinning_additive_blending
  • 基于VSCode+PlatformIO环境的ESP8266的HX1838红外模块
  • sql的性能分析
  • Linux | Uboot-Logo 修改文档(第十七天)
  • 【通讯录教程】如何将号码快速导入手机通讯录,支持苹果和安卓手机,一次性导入大量号码进入手机通讯录,基于WPF的解决方案
  • C语言中#include引用头文件的尖括号和双引号的区别
  • 情书大全v3.0.1
  • 【网络分析工具】网络工具wireshark、TCPdump、iperf使用详解