【c语言】二维数组与指针 存储原理

news2025/1/18 4:43:29

 创作不易,本篇文章如果帮助到了你,还请点赞支持一下♡>𖥦<)!! 

主页专栏有更多知识,如有疑问欢迎大家指正讨论,共同进步!

给大家跳段街舞感谢支持!ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ  

目录

一些前导知识: 

 一、二维数组在内存中的存储形态 

 二、二维数组名 地址常量用法

 1)找到 某行的首地址

 2)得到 某行的存储空间

 3)找到某行内--某列的地址

 4)得到某行 某列的存储空间

 三、&arr取整个二维数组的地址

 四、数组指针在二维数组的用法

1)利用指向一维数组的指针 遍历每一行的首地址

2)利用指向一维数组的指针 遍历每一行 每一个元素成员

3)利用指向二维数组的指针 遍历每一行 每一个元素成员


 

一些前导知识: 

利用指针变量操作数组

指针数组、数组指针

二维数组 

一、二维数组在内存中的存储形态 

我们先定义一个二维数组(以一维 的形式)

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

为什么二维数组可以这样定义呢?二维数组不是平面 表格形式的吗?

原因是二维数组在内存中的存储形式是连续的。内存中并没有“表格”。

我们可以用int指针遍历数组元素,证明二维数组在内存中是连续的: 

#include <stdio.h>
int main()
{
	/* 二维数组在内存中的存储形态 */
	int arr[3][4] = { 1,2,3,4,5,6,7,8,9,10,11,12 };   
    int* p = &arr[0][0];
	for (int i = 0; i < 3 * 4; i++)
	{
		printf("%d ", *p++);
	}
    return 0;
}

在我们使用指针遍历数组时,二维数组中的元素成行排列,证明了二维数组在内存中是连续的。

 在内存中也能查看:

 二、二维数组名 地址常量用法

二维数组数组名代表首行元素的地址,&arr为整个二维数组的地址

以下是二维数组名的用法:

1)找到 某行的首地址

printf("%d %d %d\n", arr, arr + 1, arr + 2);

 与一维数组不同的是:在二维数组中数组名代表了行地址

 2)得到 某行的存储空间

printf("%d %d %d\n", *arr, *(arr + 1), * (arr + 2));
printf("%d %d %d\n", sizeof(* arr), sizeof(* (arr + 1)), sizeof(* (arr + 2)));

 在一维数组中*arr就是得到了元素空间,取得了元素的值,而在二维数组中,*arr指得到了行的空间,并不会取到行元素的值。

在arr+1、arr+2偏移后,实际偏移了一整行的空间,所以地址相差16个字节,即一整行字节。

*arr、*arr+1、*arr+2代表的是第0行、第一行和第二行,所以字节数为16. 

 3)找到某行内--某列的地址

printf("%d %d %d\n", *(arr+1)+0, *(arr + 1)+1, *(arr + 1)+2);

 在*(arr+1)取得到了行空间后,再进行偏移就是 在此行内向列偏移,得到列地址

 4)得到某行 某列的存储空间

printf("%d %d %d\n", *(* (arr + 1) + 0), *(* (arr + 1) + 1), *(* (arr + 1) + 2));

 由于*(arr+1)+0, *(arr + 1)+1已经取得了列地址,再加星号即可取得对应列空间,即取得了该行该列元素的值。

先找行再找列的过程可以理解为:

找教室的过程:先找到楼层 (找到行) 得到了一层的空间(行空间)
然后在本层内找到房间(在行空间得到列空间)


&找到行地址 -> *得到行空间 -> &找到列地址 -> *得到列空间 !!!

在之前讲过(深度理解 [ ]运算符)  [ ] 是计算过程: 参考地址 [ 偏移量 ] ===  *(参考地址 +偏移量)  我们可以将上面的代码进行变换 将*( + )变换成  [ ]   结果让人大吃一惊:

   //得到 某行的存储空间
		printf("%d %d %d\n", *arr, *(arr + 1), *(arr + 2));
		    printf("%d %d %d\n", arr[0], arr [ 1], arr [ 2]);
   //:找到某行内--某列的地址
		printf("%d %d %d\n", *(arr + 1) + 0, *(arr + 1) + 1, *(arr + 1) + 2);
			printf("%d %d %d\n", arr [ 1] + 0, arr [ 1] + 1, arr [ 1] + 2);
   //:得到某行 某列的存储空间
		printf("%d %d %d\n", *(*(arr + 1) + 0), *(*(arr + 1) + 1), *(*(arr + 1) + 2));
			printf("%d %d %d\n", arr [ 1] [ 0], arr [ 1] [ 1], arr [ 1] [ 2]);

*到[ ]的变换过程:      *(x + v)将 *(+)都去掉 变成 x v,再在v左右加上[ ] 

在经过了*()变成 [ ]的运算后,获得了arr[ ]的形式,通过这个变换,将对二维数组理解更加深入

 arr[0], arr [ 1], arr [ 2]是 0,1,2行的存储空间,arr [ 1] + 0, arr [ 1] + 1, arr [ 1] + 2 是行内列的偏移,arr [ 1] [ 0], arr [ 1] [ 1], arr [ 1] [ 2]就得到了列的空间,同时也是数组下标法的某元素值!

三、&arr取整个二维数组的地址

printf("%d %d\n", &arr, &arr + 1); //差48 12个整数

 

 四、数组指针在二维数组的用法

1)利用指向一维数组的指针 遍历每一行的首地址

 在二维数组中,arr指得到了行的地址。

int(*ph1)[4] = arr;
for (int i = 0; i < 3; i++)
{
	printf("%d \n", ph1++);
}

 

 

2)利用指向一维数组的指针 遍历每一行 每一个元素成员

 利用指向一维数组的指针再对列偏移得到元素空间:

int(*ph2)[4] = arr;
for (int i = 0; i < 3; i++)
{
	printf("%d %d %d %d\n",*(*ph2+0),*(*ph2+1),(*ph2)[2],*(* ph2 + 3));
	ph2++;
}

 

3)利用指向二维数组的指针 遍历每一行 每一个元素成员

利用二维数组指针直接得到各元素空间: 

int(*pm)[3][4] = &arr;
for (int i = 0; i < 3; i++)
{
	for (int j = 0; j < 4; j++)
	{
	    printf("%d %d %d  ",arr[i][j],(*pm)[i][j],*(*(*pm + i)+j));
	}
	printf("\n");
}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/418903.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Adaptive Weight Assignment Scheme For Multi-task Learning

Adaptive Weight Assignment Scheme For Multi-task Learning 题目Adaptive Weight Assignment Scheme For Multi-task Learning译题用于多任务学习的自适应权重分配方案时间2022年期刊/会议IAES International Journal of Artificial Intelligence (IJ-AI) 摘要&#xff1a;如…

UDS 14229-1标准加Trace ,两万字长文详细解读

&#x1f345; 我是蚂蚁小兵&#xff0c;专注于车载诊断领域&#xff0c;尤其擅长于对CANoe工具的使用&#x1f345; 寻找组织 &#xff0c;答疑解惑&#xff0c;摸鱼聊天&#xff0c;博客源码&#xff0c;点击加入&#x1f449;【相亲相爱一家人】&#x1f345; 玩转CANoe&…

比较系统的学习 pandas(5)

pandas 常见的高级操作 1、进行复杂查询 由于不好描述&#xff0c;就举几个栗子吧&#xff0c;不明白的可以私聊我 1、pnadas 支持逻辑计算与位运算 对DataFrame的一列进行逻辑计&#xff0c;会产生一个对应的由布尔值组成的Series&#xff0c;真假值由此位上的数据 是否满…

ArrayList的深入理解

ArrayList的源码解析1. 实例化无参构造器2. add&#xff08;&#xff09;方法2.1. add&#xff08;&#xff09;List的扩容3. get&#xff08;&#xff09;方法4. remove&#xff08;&#xff09;5. modCount 的作用(面试题)5. 手写简单List实现存储删除1. 实例化无参构造器 A…

Pandas 2.0 vs Polars:速度的全面对比

前几天的文章&#xff0c;我们已经简单的介绍过Pandas 和Polars的速度对比。刚刚发布的Pandas 2.0速度得到了显著的提升。但是本次测试发现NumPy数组上的一些基本操作仍然更快。并且Polars 0.17.0&#xff0c;也在上周发布&#xff0c;并且也提到了性能的改善&#xff0c;所以我…

Apple苹果开发者Certificates, Identifiers Profiles创建

1.创建Profile: 登陆Apple开发者账号,进入开发者后台,选择Profiles 然后点击Generate a Profile生成一个配置 选择配置类型 点击继续 选择APP ID 创建APP ID 注册一个新的APP id 选择应用类型,然后点击继续

两种事件处理模式:reactor/proactor

&#xff01;&#xff01;同步通常是reactor&#xff0c;异步通常是proactor reactor 要求主线程&#xff08;I/O处理单元&#xff09;只负责监听文件描述符上是否有事件发生&#xff0c;有的话就立即将该事件通知工作线程&#xff08;逻辑单元&#xff09;&#xff0c;将 so…

SpringBoot——单元测试实践总结

文章目录单元测试概念作用黑白盒黑盒测试白盒测试逻辑覆盖1、语句覆盖2、判定覆盖3、条件覆盖4、条件/判定覆盖5、条件组合覆盖6、路径覆盖SpringBoot工程单测介绍pom依赖注意&#xff1a;Idea结构创建路径创建类和方法Controller层单测被测代码测试代码Service层单测被测代码测…

springboot集成hadoop3.2.4HDFS

前言 记录springboot集成hadoop3.2.4版本&#xff0c;并且调用HDFS的相关接口&#xff0c;这里就不展示springboot工程的建立了&#xff0c;这个你们自己去建工程很多教程。 一、springboot配置文件修改 1.1 pom文件修改 <!-- hadoop依赖 --><dependency><gro…

【从零开始学Skynet】基础篇(八):简易留言板

这一篇我们要把网络编程和数据库操作结合起来&#xff0c;实现一个简单的留言板功能。 1、功能需求 如下图所示&#xff0c;客户端发送“set XXX”命令时&#xff0c;程序会把留 言“XXX”存入数据库&#xff0c;发送“get”命令时&#xff0c;程序会把整个留言板返回给客户端。…

怎么把视频转成mp3音频,下面有四个方法

你有没有遇到过这种情况&#xff0c;看了一部电影或者纪录片&#xff0c;里面的背景音乐或者对白让你很感动&#xff0c;但是我们只需要其中的音频部分&#xff0c;比如在手机上收听音乐或者创作自己的音频内容。这时候我们可以先把视频保存下来&#xff0c;然后通过视频转音频…

光耦继电器工作原理及优点概述

光耦继电器是一种电子元器件&#xff0c;也是固态继电器的一种&#xff0c;其主要作用是隔离输入与输出电路&#xff0c;用于保护或者控制电路的正常工作。 光耦继电器工作原理是利用光电转换器将外界信号转化为光信号&#xff0c;通过光纤传输到另一端&#xff0c;再由另一端的…

【C生万物】 数组篇

欢迎来到 Claffic 的博客 &#x1f49e;&#x1f49e;&#x1f49e; 前言&#xff1a; 这个专栏好久没更新了&#xff0c;今日诗兴大发&#xff0c;打算尽快完成这个专栏&#xff0c;这期讲数组。 目录 Part1:一维数组 1.创建 2.初始化 3.使用 4.在内存中的存储 Part2:二…

安卓开发学习记录(续)

文章目录十一、综合训练&#xff08;购物车功能&#xff09;十二、内容提供者Provider十一、综合训练&#xff08;购物车功能&#xff09; 实现功能&#xff1a; 手机商品页面展示&#xff0c;加入购物车功能&#xff0c;商品详情页面&#xff0c;清空购物车&#xff0c;删除购…

C++算法初级9——递归

C算法初级9——递归 文章目录C算法初级9——递归递归求阶乘递归求斐波那契数列递归&#xff0c;简单地来说&#xff0c;就是一个函数自己调用自己。函数f()就好像是工厂中生产零件的模板&#xff0c;每次我们调用函数f()的时候&#xff0c;都会依照模板生产一个新的零件&#x…

项目4:后台管理的开发和使用(前端)

项目4&#xff1a;后台管理的开发和使用&#xff08;前端&#xff09; 1.npm包管理器的基本学习 2.利用现成后台管理系统开发 3.后台管理系统的路由配置 4.后台管理系统的地址访问配置 5.前后端联调 6.完善积分等级的前端系统 7.对前端系统的全面分析&#xff08;Vue组件…

跳槽进阿里了,其实也没那么难...

对于很多没有学历优势的人来说&#xff0c;面试大厂是非常困难的&#xff0c;这对我而言&#xff0c;也是一样&#xff0c;出身于二本&#xff0c;原本以为就三点一线的生活度过一生&#xff0c;直到生活上的变故&#xff0c;才让我有了新的想法和目标&#xff0c;因此我这个二…

【C++ -模块一 常量变量、关键字、数据类型】

C 模块一C框架代码&#xff1a;第一个C程序&#xff0c;打印hello C &#xff01;代码注释&#xff1a;一 变量和常量&#xff1a;1.1变量变量创建语法&#xff1a;1.2 常量&#xff1a;不能被修改的数据&#xff08;1&#xff09; #define定义的宏常量&#xff1a;一般写在文件…

排序(3)之交换排序

目录 前言 交换排序 1.冒泡排序 1.1冒泡排序的实现 1.2 特性总结 2.快速排序 2.1hoare版本 2.2 挖坑法 2.3 前后指针版本 3.快速排序的优化 3.1 三数取中法 3.2 小区间优化 4.快速排序的非递归实现 前言 今天小编给大家带来交换排序的内容&#xff0c;对于交换排序…

C-关键字(下)

文章目录循环控制switch-case-break-defaultdo-while-forgetchar()break-continuegotovoidvoid*returnconstconst修饰变量const修饰数组const修饰指针指针补充const 修饰返回值volatilestruct柔型数组union联合体联合体空间开辟问题利用联合体的性质,判断机器是大端还是小端enu…