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

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

相关文章

Java数组初始化的五种方式

《Java数组初始化的五种方式》数组是Java中最基础且常用的数据结构之一,其初始化方式多样且各具特点,本文详细讲解Java数组初始化的五种方式,分析其适用场景、优劣势对比及注意事项,帮助避免常见陷阱... 目录1. 静态初始化:简洁但固定代码示例核心特点适用场景注意事项2. 动态初始化:灵活但需手动管理代

go 指针接收者和值接收者的区别小结

《go指针接收者和值接收者的区别小结》在Go语言中,值接收者和指针接收者是方法定义中的两种接收者类型,本文主要介绍了go指针接收者和值接收者的区别小结,文中通过示例代码介绍的非常详细,需要的朋友们下... 目录go 指针接收者和值接收者的区别易错点辨析go 指针接收者和值接收者的区别指针接收者和值接收者的

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

shell编程之函数与数组的使用详解

《shell编程之函数与数组的使用详解》:本文主要介绍shell编程之函数与数组的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录shell函数函数的用法俩个数求和系统资源监控并报警函数函数变量的作用范围函数的参数递归函数shell数组获取数组的长度读取某下的

Python中使用正则表达式精准匹配IP地址的案例

《Python中使用正则表达式精准匹配IP地址的案例》Python的正则表达式(re模块)是完成这个任务的利器,但你知道怎么写才能准确匹配各种合法的IP地址吗,今天我们就来详细探讨这个问题,感兴趣的朋... 目录为什么需要IP正则表达式?IP地址的基本结构基础正则表达式写法精确匹配0-255的数字验证IP地

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio

C++原地删除有序数组重复项的N种方法

《C++原地删除有序数组重复项的N种方法》给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度,不要使用额外的数组空间,你必须在原地修改输入数组并在使用O(... 目录一、问题二、问题分析三、算法实现四、问题变体:最多保留两次五、分析和代码实现5.1、问题分析5.

Linux系统中配置静态IP地址的详细步骤

《Linux系统中配置静态IP地址的详细步骤》本文详细介绍了在Linux系统中配置静态IP地址的五个步骤,包括打开终端、编辑网络配置文件、配置IP地址、保存并重启网络服务,这对于系统管理员和新手都极具... 目录步骤一:打开终端步骤二:编辑网络配置文件步骤三:配置静态IP地址步骤四:保存并关闭文件步骤五:重

Linux配置IP地址的三种实现方式

《Linux配置IP地址的三种实现方式》:本文主要介绍Linux配置IP地址的三种实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录环境RedHat9第一种安装 直接配置网卡文件第二种方式 nmcli(Networkmanager command-line

Linux虚拟机不显示IP地址的解决方法(亲测有效)

《Linux虚拟机不显示IP地址的解决方法(亲测有效)》本文主要介绍了通过VMware新装的Linux系统没有IP地址的解决方法,主要步骤包括:关闭虚拟机、打开VM虚拟网络编辑器、还原VMnet8或修... 目录前言步骤0.问题情况1.关闭虚拟机2.China编程打开VM虚拟网络编辑器3.1 方法一:点击还原VM