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

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

相关文章

hdu2241(二分+合并数组)

题意:判断是否存在a+b+c = x,a,b,c分别属于集合A,B,C 如果用暴力会超时,所以这里用到了数组合并,将b,c数组合并成d,d数组存的是b,c数组元素的和,然后对d数组进行二分就可以了 代码如下(附注释): #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<que

常用的jdk下载地址

jdk下载地址 安装方式可以看之前的博客: mac安装jdk oracle 版本:https://www.oracle.com/java/technologies/downloads/ Eclipse Temurin版本:https://adoptium.net/zh-CN/temurin/releases/ 阿里版本: github:https://github.com/

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

webapp地址

F:\LSP\.metadata\.plugins\org.eclipse.wst.server.core\tmp0\wtpwebapps

【C++学习笔记 20】C++中的智能指针

智能指针的功能 在上一篇笔记提到了在栈和堆上创建变量的区别,使用new关键字创建变量时,需要搭配delete关键字销毁变量。而智能指针的作用就是调用new分配内存时,不必自己去调用delete,甚至不用调用new。 智能指针实际上就是对原始指针的包装。 unique_ptr 最简单的智能指针,是一种作用域指针,意思是当指针超出该作用域时,会自动调用delete。它名为unique的原因是这个

C语言:柔性数组

数组定义 柔性数组 err int arr[0] = {0}; // ERROR 柔性数组 // 常见struct Test{int len;char arr[1024];} // 柔性数组struct Test{int len;char arr[0];}struct Test *t;t = malloc(sizeof(Test) + 11);strcpy(t->arr,

C语言指针入门 《C语言非常道》

C语言指针入门 《C语言非常道》 作为一个程序员,我接触 C 语言有十年了。有的朋友让我推荐 C 语言的参考书,我不敢乱推荐,尤其是国内作者写的书,往往七拼八凑,漏洞百出。 但是,李忠老师的《C语言非常道》值得一读。对了,李老师有个官网,网址是: 李忠老师官网 最棒的是,有配套的教学视频,可以试看。 试看点这里 接下来言归正传,讲解指针。以下内容很多都参考了李忠老师的《C语言非

C 语言基础之数组

文章目录 什么是数组数组变量的声明多维数组 什么是数组 数组,顾名思义,就是一组数。 假如班上有 30 个同学,让你编程统计每个人的分数,求最高分、最低分、平均分等。如果不知道数组,你只能这样写代码: int ZhangSan_score = 95;int LiSi_score = 90;......int LiuDong_score = 100;int Zhou

Jenkins 插件 地址证书报错问题解决思路

问题提示摘要: SunCertPathBuilderException: unable to find valid certification path to requested target...... 网上很多的解决方式是更新站点的地址,我这里修改了一个日本的地址(清华镜像也好),其实发现是解决不了上述的报错问题的,其实,最终拉去插件的时候,会提示证书的问题,几经周折找到了其中一遍博文

计算数组的斜率,偏移,R2

模拟Excel中的R2的计算。         public bool fnCheckRear_R2(List<double[]> lRear, int iMinRear, int iMaxRear, ref double dR2)         {             bool bResult = true;             int n = 0;             dou