OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 3609|回复: 10

结构体指针问题,求解,谢谢

[复制链接]

13

主题

30

帖子

0

精华

初级会员

Rank: 2

积分
153
金钱
153
注册时间
2017-10-29
在线时间
24 小时
发表于 2018-7-28 10:56:46 | 显示全部楼层 |阅读模式
6金钱
#include "stdio.h"
typedef struct
                {
                        int m;
                        float n;
                }  A;

#define q (0X20000)
#define p ((A*) q)
int main()
{

        p->m=20;
        printf("%d",p->m);
}

1.q已经被define为常量了,为什么又可以用((A*) q)将其定义为结构体指针?
2.p是结构体指针,不指向结构体变量,为什么可以调用结构体成员?我测试了,输出结果正确,20
3.q的常量值,即0x20000,为什么是m的地址?也就是说,常量的值为什么变成了结构体成员的首地址?
4.这是C语言的什么功能,为什么没有见过?


最佳答案

查看完整内容[请看2#楼]

指针只是保存了一个地址的值,而具体的指针类型只有在访问对应的地址时才用到所以说你也可以void * p = 0x20000,((A*)p)->m = 20; 而你的应用中((A*)0x20000)->m = 20,即将0x20000为起始的地址以A的数据类型进行访问,存进去什么数据,读取出来当然是什么数据,当然这样的使用方式是要保证当前内存未被使用。正确使用方式是先申请所需的内存空间,再将指针指向该内存地址进行使用
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

19

主题

246

帖子

0

精华

高级会员

Rank: 4

积分
608
金钱
608
注册时间
2017-9-21
在线时间
171 小时
发表于 2018-7-28 10:56:47 | 显示全部楼层
本帖最后由 哆啦A萌 于 2018-7-28 11:40 编辑

指针只是保存了一个地址的值,而具体的指针类型只有在访问对应的地址时才用到所以说你也可以void * p = 0x20000,((A*)p)->m = 20;
而你的应用中((A*)0x20000)->m = 20,即将0x20000为起始的地址以A的数据类型进行访问,存进去什么数据,读取出来当然是什么数据,当然这样的使用方式是要保证当前内存未被使用。正确使用方式是先申请所需的内存空间,再将指针指向该内存地址进行使用
回复

使用道具 举报

10

主题

778

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
6745
金钱
6745
注册时间
2017-4-12
在线时间
1258 小时
发表于 2018-7-28 11:34:38 | 显示全部楼层
结构体 成员值读取用  .    结构体指针 成员值读取 用 ->
回复

使用道具 举报

31

主题

1955

帖子

3

精华

论坛元老

Rank: 8Rank: 8

积分
4523
金钱
4523
注册时间
2018-5-11
在线时间
947 小时
发表于 2018-7-28 12:40:06 | 显示全部楼层
这个是C语言最常用的功能了,知道了结构体的地址,就可以用指针访问成员了,
库函数,包括直接操作寄存器都基于C语言的这种小技巧。
typedef struct
{
  __IO uint32_t IMR;
  __IO uint32_t EMR;
  __IO uint32_t RTSR;
  __IO uint32_t FTSR;
  __IO uint32_t SWIER;
  __IO uint32_t PR;
} EXTI_TypeDef;

#define EXTI_BASE             ((uint32_t)0x40010400)       //32位常量,EXTI外设的基址
#define EXTI                ((EXTI_TypeDef *) EXTI_BASE)    //定义一个指向该基址的指针(类型为寄存器结构体)
以上这些stm32f10x.h都已经为我们定义好了,

这样就可以用EXTI->PR来访问挂起位寄存器了。
我的开源链接 https://github.com/ShuifaHe/STM32.git  请关注,点赞支持哦。
回复

使用道具 举报

82

主题

589

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1255
金钱
1255
注册时间
2017-11-18
在线时间
296 小时
发表于 2018-7-28 13:07:15 | 显示全部楼层
1.q只是一个地址值,A是结构体类型,((A*)q)是把地址值转换为结构体类型的地址,就行指针类型有char *,int *一样的
回复

使用道具 举报

82

主题

589

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1255
金钱
1255
注册时间
2017-11-18
在线时间
296 小时
发表于 2018-7-28 13:10:57 | 显示全部楼层
2.p ((A*) q)已经表明p是一个结构体类型的指针3.结构体可以类比数组来看,只是结构体的成员可以是多种类型,数组成员只能是同一种数据类型。本质都是连续的一块内存4.这个就是C语言结构体的使用
回复

使用道具 举报

31

主题

1955

帖子

3

精华

论坛元老

Rank: 8Rank: 8

积分
4523
金钱
4523
注册时间
2018-5-11
在线时间
947 小时
发表于 2018-7-28 14:15:03 | 显示全部楼层
美丽的时光机器 发表于 2018-7-28 13:07
1.q只是一个地址值,A是结构体类型,((A*)q)是把地址值转换为结构体类型的地址,就行指针类型有char *,int  ...

说得非常好,
其实q只是一个常数,它还不是地址,只不过这个常数值与我们想要的存储空间地址相等。
(A*)q就是强制类型转换,把这个常数值转换成了真正的地址,并且表明该地址处所存储的数据类型为A, A是自定义结构体类型
最后#define p ((A*) q)  就定义了p是一个指向特定地址的自定义结构体类型A的指针
其实2楼提醒得好,楼主的这个代码是有危险性的,前提是这个特定地址是空闲的,如果正好是有用数据或者更严重的,正好是代码区,程序的执行结果就不可预料了(很可能就跑飞死机了)。
我的开源链接 https://github.com/ShuifaHe/STM32.git  请关注,点赞支持哦。
回复

使用道具 举报

82

主题

589

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1255
金钱
1255
注册时间
2017-11-18
在线时间
296 小时
发表于 2018-7-28 14:34:55 | 显示全部楼层
warship 发表于 2018-7-28 14:15
说得非常好,
其实q只是一个常数,它还不是地址,只不过这个常数值与我们想要的存储空间地址相等。
(A* ...

长知识长头发了
没有脑袋
回复

使用道具 举报

31

主题

1955

帖子

3

精华

论坛元老

Rank: 8Rank: 8

积分
4523
金钱
4523
注册时间
2018-5-11
在线时间
947 小时
发表于 2018-7-28 14:49:10 | 显示全部楼层

哈哈,谦虚了。
另外,再多说几句解释一下,
之所以说这种用法有危险,诚如2楼所说,没有事先申请内存。
楼主的代码始终没有声明一个变量实体,相当于临时访问一片内存区。

而我4楼所举的例子(这是在ARM官方库函数中广泛使用的)虽然几乎和楼主一模一样,
也没有声明变量实体,但这个特定地址就是芯片寄存器的实际物理地址,
对此地址进行读写正好达到我们访问该寄存器的需求。
我的开源链接 https://github.com/ShuifaHe/STM32.git  请关注,点赞支持哦。
回复

使用道具 举报

1

主题

56

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
265
金钱
265
注册时间
2017-10-11
在线时间
79 小时
发表于 2018-7-28 14:57:58 | 显示全部楼层
很好,不错。
回复

使用道具 举报

0

主题

4

帖子

0

精华

初级会员

Rank: 2

积分
55
金钱
55
注册时间
2018-9-28
在线时间
11 小时
发表于 2018-10-15 12:53:20 | 显示全部楼层
warship 发表于 2018-7-28 14:49
哈哈,谦虚了。
另外,再多说几句解释一下,
之所以说这种用法有危险,诚如2楼所说,没有事先申请内存 ...

讲得很透彻,谢谢!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-20 22:58

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表