注册 登录
编程论坛 C语言论坛

想不明白,求解释一下

小野菊的夏天 发布于 2020-07-03 00:02, 2365 次点击
程序代码:
#include<stdio.h>
#define MaxSize 100

typedef struct {
    int a[MaxSize];
    int length;   
} Sqlist;

int main(){
    Sqlist L;
    Sqlist *P = &L;
    for(int i=0;i<50;i++){
        L.a[i]=i;
    }
    (*P).length = 50;

    printf("%d\n",P);
    printf("%d\n",*P);
    printf("%d\n",L);
    printf("%d\n",L.a);
    printf("%d\n",&L.a);
   
    return 0;   
}

在学数据结构的时候突然无聊试了下用%d,直接输出链表L看看是什么东西,发现指针P里面是L.a[0]的地址,而用*P直接进行输出与L直接输出的值是一样的这是为什么,P所指的不应该是L.a[0]的值吗?
7 回复
#2
fulltimelink2020-07-03 10:00
不知道这个算不算链表
p指向的是L的地址, 而a[0] 恰巧是L的第一个元素,所以L的地址和a[0]的地址一样
*P =L 所以 *P和L强制转换为int, 输出 结果 是一样的
#3
lin51616782020-07-03 10:16
*p 是一个Sqlist
你实际传入printf的是一个完整的结构体
可以接近等效于传入了int [MaxSize]和int
假如你多写一个 %d 大部分实现中不做特殊设置 你可以看到数组第二个元素
只有本站会员才能查看附件,请 登录



备注 这段代码仅用于说明参数内存布局的一种可能
仅供参考 实际上这样的做法是未定义行为 不可依赖
#4
八画小子2020-07-03 10:17
首先,
程序代码:
printf("%d\n",P);
    printf("%d\n",*P);
    printf("%d\n",L);
    printf("%d\n",L.a);
    printf("%d\n",&L.a);

这五行在编译的时候会发生有警告,我们不能简简单单看到能编译得到东西就觉得代码写得合适,而是应该把所有的警告也给消除了。因为何编译错误比起来,警告带来的错误可能更隐晦,更难定位。

下面具体说几行代码吧:
1、P是指针,所以即便你想打印P的值,那也必须用printf("%p\n",P);,如果想用%d,那么就需要先讲指针强转为对应的32位(或64位,具体看运行平台的位数)无符号数。
2、P是指向Sqlist类型的指针,P具体指向L变量。所以,*P和L其实是同一个东西。所以,printf("%d\n",*P);和printf("%d\n",L);是一样的,且逻辑上这么写是错误的,因为不能讲结构体类型直接转换为整型而不发生变化。
3、L.a是数组,&L.a是数组指针,都不能直接转换为整形。不要细究结果为什么一样,你只要知道这么些得不到你想要的结果即可。具体还要看具体编译器,因为某些编译器上是编译不通过的。
4、(*P).length = 50;这种写法太罗嗦,可以换用P->length = 50;
#5
lin51616782020-07-03 10:18
同理 你输出L 也是传入一个完整的结构体 接近于传入 MaxSize+1个int
写2个%d 可以看到 数组前2个元素
#6
小野菊的夏天2020-07-03 14:24
回复 5楼 lin5161678
我用的Devcpp 编译,*p出来不是数组的第一个值,的是一串数字地址,可能不同编译器具体实现不一样多谢🙏
#7
小野菊的夏天2020-07-03 14:25
回复 4楼 八画小子
懂了,多谢🙏
#8
lin51616782020-07-03 16:00
以下是引用小野菊的夏天在2020-7-3 14:24:32的发言:

我用的Devcpp 编译,*p出来不是数组的第一个值,的是一串数字地址,可能不同编译器具体实现不一样多谢🙏

也存在这种情况
因为你的结构体对象太大
借助指针实现传递结构体对象
入栈指针 打印指针

总之 %d 输出一个结构体对象是一个未定义行为
不要依赖
1