图解数组指针与多维数组(附:为什么指针加一,地址不一定加一)

2024-08-24 07:32

本文主要是介绍图解数组指针与多维数组(附:为什么指针加一,地址不一定加一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这里不是单纯讨论什么是数组指针,什么是指针数组,而是在掌握了一些知识后再回头看看数组指针与数组到底怎么理解。(数组指针:指向数组的指针。指针数组:指针构成的数组)

 先放上一道题:


答案是10,20,30。


虽然是很常见的题,对于一个刚开始学C语言可能就可以做出来,但是你确定你真正的理解了么?当你学的多的时候可能就会有其他疑问了。

比如我知道int(*p)[3]中的p是数组指针,也就是p是一个指向长度为3的一个数组的指针。那么我们看看这个指针都可以怎么去用。

我们从基本的一维数组去考虑,这样p指向的只能是长度为3的数组比如

int m[6]={1,2,3};//错误
int m[3]={1,2,3};
p=&m;
cout<<p[0][0]<<","<<p[0][1]<<","<<p[0][2]<<","//打印1,2,3
<<*(m+1)//打印2
<<*(p[0]+1)//打印2
<<*(p[1]+1)<<endl;//打印-858993460,这个数不同机器上,不同情况值是不同的
<<m<<","//打印002FFD04
<<&m[0]<<","//打印002FFD04
<<&m[0]+1<<","//打印002FFD08
<<&m[1]<<endl;//打印002FFD08


这里的p[0][1]看起来可能有点奇怪,不过这恰恰能说明二维数组到底是怎么回事

p是一个指向数组m的指针,那么p[0]其实就是对p的一个解引用,也就是等价于*p,也等价于m

所以p[0][0]也就是m[0]。

这样顺理成章的我们来看看二维数组

int n[][3]={10,20,30,40,50,60};
int (*p)[3];
p=n; 
cout<<p[0][0]<<","<<p[0][1]<<","<<p[1][0] //打印10,20,40
<<","<<*(p[1]+1) //打印50
<<","<<(*p)[2]<<","<<(*p+1)[2]<< //打印30,40
","<<p[0]<<","<<p[1]<<endl; //打印003AFEF0,003AFEFC
cout<<","<<(p+1)[0]<<","<<n[1]<< "," <<*n[1]<< endl; //打印003AFEFC, 003AFEFC,40

就如前面所说的,这次p是一个指向数组n的指针,那么p[0]其实就是对p的一个解引用,也就是等价于*p,也等价于*n,所以p[0][0]也就是n[0][0]。

 

 

你可能发现前面多次打印了地址,这个还能说明什么呢?我们发现对指针地址进行加一后,我们的地址并不会直接加一,而是加了一个中间包含数据所占字节数的大小。也就是说,如果指针指向一个int型数据(一般机器上int型都是4个字节),那么加一操作就会在地址上加4,比如上面的&m[0]与&m[0]+1。如果是指向char,加一操作就会在地址上加1。而如果对指向二维数组头的指针加一,那么这个地址就会加上一行数据的总字节数大小比如上面的p[0]与p[1]就分别等价于p与p+1。

 

看到这里可能还会有疑惑?我让int型的指针执行加一操作时地址也加一不可以么?

我是这样理解的,按照现在的计算机语言情况,最小的数据类型就是单字符,其长度为一个字节。所以你的指针大小至少要能保证指向每一个字节的内容,这样才能保证所有的数据都可以被取到。所以指针可以在加一的同时地址也加一,也就是每个字节对应一个地址。但是当你指向int类型的时候,你的加一操作也是让地址加一的话,你指向的数据就变成了int型第二个字节所组成的数,也就是说你无法获取完整的int类型数据。

 

最后附上一张地址逻辑的示意图,有助于大家理解。



这篇关于图解数组指针与多维数组(附:为什么指针加一,地址不一定加一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1101836

相关文章

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Ubuntu固定虚拟机ip地址的方法教程

《Ubuntu固定虚拟机ip地址的方法教程》本文详细介绍了如何在Ubuntu虚拟机中固定IP地址,包括检查和编辑`/etc/apt/sources.list`文件、更新网络配置文件以及使用Networ... 1、由于虚拟机网络是桥接,所以ip地址会不停地变化,接下来我们就讲述ip如何固定 2、如果apt安

查询SQL Server数据库服务器IP地址的多种有效方法

《查询SQLServer数据库服务器IP地址的多种有效方法》作为数据库管理员或开发人员,了解如何查询SQLServer数据库服务器的IP地址是一项重要技能,本文将介绍几种简单而有效的方法,帮助你轻松... 目录使用T-SQL查询方法1:使用系统函数方法2:使用系统视图使用SQL Server Configu

使用Java实现获取客户端IP地址

《使用Java实现获取客户端IP地址》这篇文章主要为大家详细介绍了如何使用Java实现获取客户端IP地址,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 首先是获取 IP,直接上代码import org.springframework.web.context.request.Requ

Linux环境变量&&进程地址空间详解

《Linux环境变量&&进程地址空间详解》本文介绍了Linux环境变量、命令行参数、进程地址空间以及Linux内核进程调度队列的相关知识,环境变量是系统运行环境的参数,命令行参数用于传递给程序的参数,... 目录一、初步认识环境变量1.1常见的环境变量1.2环境变量的基本概念二、命令行参数2.1通过命令编程

C++实现获取本机MAC地址与IP地址

《C++实现获取本机MAC地址与IP地址》这篇文章主要为大家详细介绍了C++实现获取本机MAC地址与IP地址的两种方式,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 实际工作中,项目上常常需要获取本机的IP地址和MAC地址,在此使用两种方案获取1.MFC中获取IP和MAC地址获取

C/C++通过IP获取局域网网卡MAC地址

《C/C++通过IP获取局域网网卡MAC地址》这篇文章主要为大家详细介绍了C++如何通过Win32API函数SendARP从IP地址获取局域网内网卡的MAC地址,感兴趣的小伙伴可以跟随小编一起学习一下... C/C++通过IP获取局域网网卡MAC地址通过win32 SendARP获取MAC地址代码#i

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

龙蜥操作系统Anolis OS-23.x安装配置图解教程(保姆级)

《龙蜥操作系统AnolisOS-23.x安装配置图解教程(保姆级)》:本文主要介绍了安装和配置AnolisOS23.2系统,包括分区、软件选择、设置root密码、网络配置、主机名设置和禁用SELinux的步骤,详细内容请阅读本文,希望能对你有所帮助... ‌AnolisOS‌是由阿里云推出的开源操作系统,旨