指针问题

flcnrcn 2018-12-03 03:57:19

//函数声明
void function( int arr[2][3][4]);
..................................
int array[2][3][5] = {...};
int (*q)[2][3][5] = &array;
//函数调用
function(*q);

这里实参为*q好理解因为*q的值就是array,但是我用printf("array=%x q=%x\n",array, q);打印出来的值竟然相同。
问题:q的值应该是array这个地址存在于内存中的地址(换句话说q这块内存用来存放array),而array的值是数组首元素的地址,
他俩在内存的位置应该是不一样的怎么打印出来的值会一样呢?
...全文
336 9 打赏 收藏 转发到动态 举报
写回复
用AI写文章
9 条回复
切换为时间正序
请发表友善的回复…
发表回复
架构师李肯 2018-12-22
  • 打赏
  • 举报
回复
楼主的代码:
//函数声明
void function( int arr[2][3][4]);
..................................
int array[2][3][5] = {...};
int (*q)[2][3][5] = &array;
//函数调用
function(*q);

实际上这个 q是一个数组指针,但它相当于一个4维数组了,而不是一个3维数组,而 int (*q)[2][3][5] = &array; 实际上就等同于 q = &array = array(它也是表示array数组的地址);
所以得到输出,array 和 q的值是一样的。

3维数组不好理解,我们先用一个2维数组捋一捋;
假设有一个2维数组, int array[2][3];
我们定义一个数组指针并指向 array,我们会这样写 int (*q)[3] = array; 注意这里是没有[2]
那么这个时候,q就等同于array这个2维数组,q 还是等于 array (数组的首地址);测试如下:
{
int array[2][3];
int (*p)[3] = array;

//输出结果4个值都是一样的
printf("%d, %d; %d, %d\n", p, array, *p, &array);
//1453427088, 1453427088; 1453427088, 1453427088
}


回到楼主的定义,对比下就知道原因。分析复杂的问题,建议先由简单点的问题入手,逐步对比,提升理解。
卟倁檤、 2018-12-06
  • 打赏
  • 举报
回复
指针问题从来就做不对
赵4老师 2018-12-06
  • 打赏
  • 举报
回复
电脑内存或文件内容或传输内容只是一个一维二进制字节数组及其对应的二进制地址;
人脑才将电脑内存或文件内容或传输内容中的这个一维二进制字节数组及其对应的二进制地址的某些部分看成是整数、有符号数/无符号数、浮点数、复数、英文字母、阿拉伯数字、中文/韩文/法文……字符/字符串、汇编指令、函数、函数参数、堆、栈、数组、指针、数组指针、指针数组、数组的数组、指针的指针、二维数组、字符点阵、字符笔画的坐标、黑白二值图片、灰度图片、彩色图片、录音、视频、指纹信息、身份证信息……

计算机组成原理→DOS命令→汇编语言→C语言(不包括C++)、代码书写规范→数据结构、编译原理、操作系统→计算机网络、数据库原理、正则表达式→其它语言(包括C++)、架构……

对学习编程者的忠告:
多用小脑和手,少用大脑、眼睛和嘴,会更快地学会编程!
眼过千遍不如手过一遍!
书看千行不如手敲一行!
手敲千行不如单步一行!
单步源代码千行不如单步Debug版对应汇编一行!
单步Debug版对应汇编千行不如单步Release版对应汇编一行!
不会单步Release版对应汇编?在你想单步Release版C/C++代码片断的前面临时加一句DebugBreak();重建所有,然后在IDE中运行。(一般人我不告诉他!

VC调试时按Alt+8、Alt+7、Alt+6和Alt+5,打开汇编窗口、堆栈窗口、内存窗口和寄存器窗口看每句C对应的汇编、单步执行并观察相应堆栈、内存和寄存器变化,这样过一遍不就啥都明白了吗。
对VC来说,所谓‘调试时’就是编译连接通过以后,按F10或F11键单步执行一步以后的时候,或者在某行按F9设了断点后按F5执行停在该断点处的时候。

http://edu.csdn.net/course/detail/2344 C语言指针与汇编内存地址-一.代码要素
赵4老师 2018-12-05
  • 打赏
  • 举报
回复
理解讨论之前请先学会如何观察
//char (*(*x[3])())[5];//x是什么类型的变量?
//
//分析C语言声明,关键是搞清楚这个变量是个什么东西(函数、指针、数组),
//是函数那么剩下的就是他的参数和返回值,
//是指针那剩下部分是说明他指向什么,
//是数组剩下的部分就是说明数组的成员是什么类型。
//解析C语言声明规则:
//从左侧第一个标识符开始,按照优先级进行结合。*表示是..的指针,const表示只读的,volatile表示可变的,[]表示是数组,()表示是函数。
//
//x和[3]结合说明是一个大小为3的数组,该数组的每个元素为一类指针,该类指针指向一类函数,该类函数无参数,返回一类指针,该类指针指向一个大小为5的char型数组
#include <stdio.h>
#include <typeinfo.h>
char num[5];
char (*x00())[5] {
return #
}
int main() {
char (*(*x[3])())[5];//是个数组,大小为3
char (*(*x0 )())[5];//数组的元素,是个函数指针
char (*( x00 )())[5];//函数原型,参数为空,返回值为指针
char (* x000 )[5];//返回值

x0 = x00;
x[0] = x0;
x[1] = x0;
x[2] = x0;
printf("typeid(x).name() is %s\n",typeid(x).name());
return 0;
}
//typeid(x).name() is char (* (__cdecl**)(void))[5]
flcnrcn 2018-12-05
  • 打赏
  • 举报
回复
引用 3 楼 自信男孩 的回复:
指针变量p和array的地址不一样,你可以打印一下&p和array的地址看一下。但是指针变量里存放的是地址,这个地址就是array的地址。因此p的值和&array和array是一样的。
&array和array是一样的,那&array怎么理解。
qlzhj 2018-12-05
  • 打赏
  • 举报
回复
q是不带*的表示的是那一段内存空间,里边存储的内容是arr地址,所以打印结果一样。我的理解
flcnrcn 2018-12-05
  • 打赏
  • 举报
回复
引用 6 楼 赵4老师 的回复:
理解讨论之前请先学会如何观察
//char (*(*x[3])())[5];//x是什么类型的变量?
//
//分析C语言声明,关键是搞清楚这个变量是个什么东西(函数、指针、数组),
//是函数那么剩下的就是他的参数和返回值,
//是指针那剩下部分是说明他指向什么,
//是数组剩下的部分就是说明数组的成员是什么类型。
//解析C语言声明规则:
//从左侧第一个标识符开始,按照优先级进行结合。*表示是..的指针,const表示只读的,volatile表示可变的,[]表示是数组,()表示是函数。
//
//x和[3]结合说明是一个大小为3的数组,该数组的每个元素为一类指针,该类指针指向一类函数,该类函数无参数,返回一类指针,该类指针指向一个大小为5的char型数组
#include <stdio.h>
#include <typeinfo.h>
char num[5];
char (*x00())[5] {
    return #
}
int main() {
    char (*(*x[3])())[5];//是个数组,大小为3
    char (*(*x0  )())[5];//数组的元素,是个函数指针
    char (*( x00 )())[5];//函数原型,参数为空,返回值为指针
    char (*  x000   )[5];//返回值

    x0 = x00;
    x[0] = x0;
    x[1] = x0;
    x[2] = x0;
    printf("typeid(x).name() is %s\n",typeid(x).name());
    return 0;
}
//typeid(x).name() is char (* (__cdecl**)(void))[5]
您好! printf("typeid(arr).name() is %s\n",typeid(arr).name());---它是int [2][3][5] printf("typeid(&arr).name() is %s\n",typeid(&arr).name());---它是int (*)[2][3][5] printf("typeid(q).name() is %s\n",typeid(q).name());---它是int (*)[2][3][5] printf("typeid(&q).name() is %s\n",typeid(&q).name());---它是int (**)[2][3][5] printf("arr=%x &arr=%x q=%x &q=%x\n",arr, &arr, q, &q);---前三项的值相同 问题:arr, &arr打印出来的值一样这里理解不了。因为我认为&arr存放的是arr的地址是两块不同的内存地址当然不一样, 后面q, &q地址不一样就是证明。求赐教。。。
自信男孩 2018-12-04
  • 打赏
  • 举报
回复
指针变量p和array的地址不一样,你可以打印一下&p和array的地址看一下。但是指针变量里存放的是地址,这个地址就是array的地址。因此p的值和&array和array是一样的。
zhouqunhai 2018-12-04
  • 打赏
  • 举报
回复
当然是相等的,步长不一样
英雄@末路 2018-12-04
  • 打赏
  • 举报
回复
数组和指针在去地址符作用下,结果不一样。
其中数组本身和其取址符下的地址是一样的(数组可以理解成别名)。
int a[8],
打印 "%p",a == "%p",&a
云山大侠 2018-12-03
  • 打赏
  • 举报
回复
是这样的,array是数组的首地址,q是数组的首地址,当然是相等的,不相等是的&q和&array

69,371

社区成员

发帖
与我相关
我的任务
社区描述
C语言相关问题讨论
社区管理员
  • C语言
  • 花神庙码农
  • 架构师李肯
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧