C语言数组——多维数组
创始人
2024-06-02 07:17:00
0

本节讲解下多维数组的一些易错点和 超过2维后的数组简单认识

 

多维数组的空间想象

  • 一维数组:一列长表或一个向量
  • 二维数组:一个表格或一个平面矩阵
  • 三维数组:一本书或三维空间的一个方阵
  • 多维数组: 多维空间的一个数据列阵

在这里插入图片描述
 


C语言在内存中的存放情况

在计算机内存中,一个数组通常是一块连续的内存空间,它按照一定的顺序存储数组元素。数组中的每个元素都有一个唯一的索引,可以用来访问该元素。

具体来说,数组在内存中的存放方式通常是:

数组元素按照定义的顺序在内存中依次排列,每个元素占据一个连续的内存单元。数组的起始地址即为第一个元素的内存地址,而后续元素的地址则按照元素类型所占用的字节数进行连续的地址递增。数组的元素可以通过下标访问,下标可以理解为偏移量,即相对于数组首地址的偏移量。对于多维数组,内存中的存储方式与一维数组类似,只不过多维数组的元素可能需要多个下标来访问。

需要注意的是,不同的编程语言可能对数组的实现方式有所不同,但是大多数语言都采用了类似的内存存储方式。同时,由于内存空间有限,当数组元素过多时,可能会导致内存溢出等问题。

举例:二维数组在内存按行列存储

  • int a[3][2];3 行2列,6个元素表示1个3行2列的矩阵
    Alt

 


❗易错点:二维数组初始化不能缺少列数,可以省略行

定义接受二维数组作为实参的形参时,定义时可以省略二维数组的行数,但不能省略列数。

简单说查找时候,a[3][4]说明是第三行第四列,知道二维数组是a[][4],知道列数,就可以查询了

a[3][4],知道列是4,直接锁定a[3][4]的地址

因为数组的内存布局是 按照行优先的顺序进行的。也就是说,每一行的元素是连续存储的,而每一列的元素则是间隔存储的,如果不指定列数,编译器无法确定每个元素之间的间隔,也就无法正确地计算出数组中元素的地址。

Alt

比如:

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

这里指定了每行有3个元素,编译器就可以根据行数和列数计算出每个元素的地址。

 


三维数组熟悉

在这里插入图片描述
无非是变成x,y,z面三维度了

这里 a[][][] , 后面二个还是xy面,也就是二维数组的行列

三维数组的初始化

//三维数组
int a[2][3][4] =
{{{6,2,3,4},{5,11,15,8},{8,20,10,11}},{{0,0,3,4},{5,0,7,8},{8,1,18,31}}

这里插入下一个重要的知识点:

‼️ 几维数组指针

例子:

关于几维数组指针:
a[i][j][k][2] 等同于 *(*(*(*(a+i)+j)+k)+2) ?
·

维度太高不好理解,以数组int a[i+1][j+1]二维数组为例。
如何取得最后一个元素a[i][j]的值? a=*a,即该数组的首地址,亦即第0行的首地址(如果要取第0行的第0个元素,即a[0][0],可用**a*(a+i)表示第i行首地址,*(a+i)+j第i行第j列的地址。而*表示引用地址的存储内容。
故*(*(a+i)+j)表示i行j列的元素取值。 所以更高维同理。 

—— 这里解释下为什么引用是a?**

对于二维数组来说,先通过a来获取第一维的指针,然后再通过这个指针来获取第二维的元素。

我们不能直接改变访问里面某个值,举例说明:

假设我们定义了一个二维数组a:

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

如果我们想要访问数组a中的某个元素,例如a[1][2],可以直接赋值

int value = a[1][2];  // 直接访问数组元素

也可以使用如下指针:

int **p = a; //指向第一行的指针 = a[0]
int value = *(*(p+1)+2);  // 使用指针的指针访问数组元素

p是一个指向指针的指针,它指向了数组a的第一行的指针,也就是指向a[0]的指针。

因此,p+1指向了数组a的第二行的指针,也就是指向a[1]的指针。接着,*(p+1)表示第二行的指针,即a[1],再加上偏移量2,就可以访问a[1][2]这个元素了。

最后直接把它元素值给value值


双重指针的解引用操作 * 在这个表达式中出现了两次。我们可以将它们分开来看:

  1. 首先是 *(p+1),这里的 p+1 表示将指针 p 的值加上1,也就是让指针指向二维数组的第2行。然后再使用解引用操作符 *,表示获取指向第2行的指针的值,即第2行的首个元素的地址。

  2. 接着是 *(*(p+1)+2),这里的 *(p+1) 表示获取指向第2行的指针的值,即第2行的首个元素的地址。然后再加上2,表示指向该行第3列的元素的地址。最后再使用解引用操作符 *,表示获取该元素的值。


 

三维数组例子

#include "stdio.h"
void main(){//三维数组int a[2][3][4] ={{{6,2,3,4},{5,11,15,8},{8,20,10,11}},{{0,0,3,4},{5,0,7,8},{8,1,18,31}}};int i, j, k;int(*p)[3][4];					//p是指向二维数组的指针 p = a;							//p指向三维数组的0行 (也即p指向一个二维数组) //p=&a[0];						//与上等价printf("				【三维数组的遍历】:\n\n");for (i = 0; i < 2; i++) {for (j = 0; j < 3; j++) {for (k = 0; k < 4; k++) {
//元素值遍历 :  *(*(*(p + i) + j) + k相当于 *a[i][j][k]printf("a[%d][%d][%d]=%-9d  ", i, j, k, *(*(*(p + i) + j) + k));//元素值 printf("a[%d][%d][%d]=%-9d  ", i, j, k, *(*(a[i] + j) + k));	//元素值 printf("a[%d][%d][%d]=%-9d  ", i, j, k, *(a[i][j] + k));		//元素值 printf("a[%d][%d][%d]=%-9d  ", i, j, k, *(&a[i][j][k]));		//元素值 printf("a[%d][%d][%d]=%-9d  ", i, j, k, a[i][j][k]);			//元素值 printf("a[%d][%d][%d]=%-9d\n", i, j, k, p[i][j][k]);			//元素值 //元素地址遍历: 
//				printf("&a[%d][%d][%d]=%-9d  ", i, j, k, *(*(p + i) + j) + k);	//元素地址 
//				printf("&a[%d][%d][%d]=%-9d  ", i, j, k, *(a[i] + j) + k);		//元素地址
//				printf("&a[%d][%d][%d]=%-9d  ", i, j, k, a[i][j] + k);			//元素地址
//				printf("&a[%d][%d][%d]=%-9d\n", i, j, k, &a[i][j][k]);			//元素地址//由于少了整体括号,所以出现错误的遍历,只有a[0][0][0],a[0][1][0],a[0][2][0],a[1][0][0],a[1][1][0],a[1][2][0]正确
//
//				printf("a[%d][%d][%d]=%-9d  ", i, j, k, **(*(p + i) + j) + k);	//错误,由于最外层少括号 
//				printf("a[%d][%d][%d]=%-9d  ", i, j, k, **(a[i] + j) + k);		//错误,由于最外层少括号 
//				printf("a[%d][%d][%d]=%-9d  ", i, j, k, *a[i][j] + k);			//错误,由于最外层少括号 
//				printf("a[%d][%d][%d]=%-9d\n", i, j, k, *&a[i][j][k]);			//正确 }printf("\n");}}
}【【【注意】】】:在对地址取*运算时, 一定要加“()”,也即对地址整体取“括号”,然后*,否则将得不到期望值!
比如:*(a[i] + j) + k表示 & a[i][j][k]  ====> 也即元素a[i][j][k]的地址。对其取*运算应该是*(*(a[i] + j) + k) ====> 表示是元素a[i][j][k]的值。而不能写成这样:**(a[i] + j) + k ====> 并不是我们期望的各元素值!当然了:对于&a[i][j][k]求其元素值时,可以对其直接取*,不必在意()整体的问题!【分析如下】:
for (i = 0; i < 2; i++) {for (j = 0; j < 3; j++) {for (k = 0; k < 4; k++) {printf("a[%d][%d][%d]=%-9d  ", i, j, k, **(a[i] + j) + k);		//**(*(p + i) + j) + k, *a[i][j] + k同理}}
}【输出结果】:[6],7,8,9  [5],6,7,8  [8],9,10,11  [0],1,2,3  [5],6,7,8  [8],9,10,11
【很明显】:只有a[0][0][0], a[0][1][0], a[0][2][0], a[1][0][0], a[1][1][0], a[1][2][0]正确,其他值并不是个元素的值!
【假如】:i=0, j=0, k=0; ==> **(a[i]+j)+k ===> **(a[0]+0)+0 ===> *(*(a[0]+0))+0 ===> *(a[0][0])+0 ===>*(a[0][0])+0 ===> *(&a[0][0][0])+0= 6i=0, j=0, k=1; ==> **(a[i]+j)+k ===> **(a[0]+0)+1 ===> *(*(a[0]+0))+1 ===> *(a[0][0])+1 ===>*(a[0][0])+1 ===> *(&a[0][0][0])+1= 7i=0, j=0, k=2; ==> **(a[i]+j)+k ===> **(a[0]+0)+2 ===> *(*(a[0]+0))+2 ===> *(a[0][0])+2 ===>*(a[0][0])+2 ===> *(&a[0][0][0])+2= 8i=0, j=0, k=3; ==> **(a[i]+j)+k ===> **(a[0]+0)+3 ===> *(*(a[0]+0))+3 ===> *(a[0][0])+3 ===>*(a[0][0])+3 ===> *(&a[0][0][0])+3= 9
其他情况亦是如此!

节选自:三维数组


相关内容

热门资讯

全红婵18岁后首秀 跳水世界杯... 来源:央视体育 2025跳水世界杯瓜达拉哈拉站女子单人10米台预赛,全红婵表现出色获得第一名,顺利晋...
Aptos 启动“全球交易引擎... 吴说获悉,Aptos 宣布启动“全球交易引擎”计划,旨在构建支持稳定币、RWA、链上订单簿和跨链账户...
4月4日生意社白糖基准价为62... 生意社04月04日讯 4月4日,生意社白糖基准价为6201.67元/吨,与本月初(62...
杨柳飞絮要来了,清明假期仍有大... 北京市气象台发布清明节期间(4月4日至6日)天气预报,预计北京天气以晴到多云为主,能见度总体较好。受...
清明缅怀水利先辈,我与“豆包”... 小编豆包,你好。今天是中国的传统节日清明节。我要做一篇关于清明节的推送,来找你找找灵感。豆包清明节是...
陇电入鲁工程送受两端启动带电联...   原标题:陇电入鲁工程送受两端启动带电联调联试中国甘肃网4月4日讯 据甘肃日报报道(新甘肃·甘肃日...
云南省发布省管干部任前公示公告 中共云南省委组织部云干任公示〔2025〕4号省管干部任前公示公告  根据《党政领导干部选拔任用工作条...
家国永念 精神永续 转自:人民日报清明又至,狼牙山上桃花开得正艳,刘老庄82棵青松擎天而立,乌斯浑河静静流淌,四行仓库西...
有一本书的主人公叫孟飞,讲的是... 有一本书的主人公叫孟飞,讲的是在2005年被闪电劈到了1999年之后开发游戏发家了!内是一部校园内形...
观微知著是成语吗? 观微知著是成语吗?观微知著同见微知著,是成语。【拼音】:jiàn wēi zhī zhù【解释】:微...
种田一定没出息吧 种田一定没出息吧我们班有辩论赛,这是辩题感谢潜水艇's husband多少种田的父母把自己的子女培养...
《火凤燎原》是根据《三国策》以... 《火凤燎原》是根据《三国策》以及其他文献写的还是根据《三国演义》的野史写的。还是作者自己的想象?火凤...
我怎么打好篮球 我怎么打好篮球有点瘦,最好加点体重。如果要练弹跳的话,天天蛙跳,下蹲,虽然是最简单的,但是也是最好的...
别人说我嘴有毒我要怎么怼回去 别人说我嘴有毒我要怎么怼回去如果是异性的话,你说,你中毒了没?中了我的毒就是我的人了。知道就好 不...
3月中国物流业景气指数51.5... 来源:央视新闻客户端  中国物流与采购联合会今天(3日)公布3月份中国物流业景气指数。在需求普遍回升...
一季度新疆霍尔果斯口岸出入境人... 中新网新疆霍尔果斯4月4日电 (李明 胡嘉琛)据新疆霍尔果斯出入境边防检查站统计,今年第一季度,霍尔...
《火影忍者》佐助的实力巅峰期是... 《火影忍者》佐助的实力巅峰期是哪个阶段?在得到六道仙人之后,开启了六勾玉轮回眼,这个时候实力是最高的...
4月4日生意社棕榈油基准价为9... 生意社04月04日讯 4月4日,生意社棕榈油基准价为9800.00元/吨,与本月初(9...
离谱!美国“对等关税”闹剧连企... 近日,美国宣布对贸易伙伴设立10%的最低基准关税,并对某些贸易伙伴征收更高关税。据悉,印度洋上的赫德...
日股跌幅扩大,日本东证指数下跌... 更多消息,持续更新中日股跌幅扩大,日本东证指数下跌3%。瑞穗金融一度下跌10%创2024年8月以来最...