如何使用union处理浮点数据?

新闻资讯   2023-06-30 12:02   75   0  

一、联合体的使用和分析

1、联合体

联合体(union)与结构体(struct)有一些相似之处。但两者有本质上的不同。在结构体中,各成员有各自的内存空间,一个结构变量的总长度是各成员长度之和。而在联合体中,各成员共享一段内存空间,一个联合变量的长度等于各成员中最长的长度。应该说明的是,这里所谓的共享不是指把多个成员同时装入一个联合变量内,而是指该联合变量可被赋予任一成员值,但每次只能赋一种值,赋入新值则冲去旧值。

2、声明共用体类型

一个联合体类型必须经过定义之后,才能使用它,才能把一个变量声明定义为该联合体类型。

联合变量的声明和结构变量的声明方式相同,也有三种形式。

一般形式具体如下:

union 共用体名
{
   //成员列表
   数据类型 成员名;
   数据类型 成员名;
   ....
}

示例:

union _MQ
{
 char data8[4];
 int  data32;
}; 

3、定义共用体变量

一般形式具体如下:

union _MQ
{
 char data8[4];
 int  data32;
}; 
//定义共用体变量
union _MQ mq;

合并简化形式具体如下:

union _MQ
{
 char data8[4];
 int  data32;
}mq;

匿名结构体形式具体如下:

union 
{
 char data8[4];
 int  data32;
}mq;

使用 typedef 声明共用体类型,再定义共用体变量:

typedef union MQ
{
 char data8[4];
 int  data32;
} _MQ;
//定义共用体变量,以下两种效果一样
union MQ mq;
_MQ mq;

4、上代码:

#include <stdio.h>
  
union Test {
        unsigned char a;
        unsigned short b;
        unsigned int c;
};
 
int main(void)
{
        union Test test;
        printf("%lu\n", sizeof(union Test));
        printf("%lu\n", sizeof(test));
        printf("&test = %p, &test.a = %p, &test.b = %p, &test.c = %p\n", &test, &test.a, &test.b, &test.c);
        test.c = 0x11223344;
        printf("test.c = %x\n", test.c);
        printf("test.a = %x\n", test.a);
        printf("test.b = %x\n", test.b);
        test.a = 0x88;
        printf("test.c = %x\n", test.c);
        printf("test.a = %x\n", test.a);
        printf("test.b = %x\n", test.b);
        return 0;
}

结果:

4
4
&test = 0x7fff4a0708c4, &test.a = 0x7fff4a0708c4, &test.b = 0x7fff4a0708c4, &test.c = 0x7fff4a0708c4
test.c = 11223344
test.a = 44
test.b = 3344
test.c = 11223388
test.a = 88
test.b = 3388

结果分析:

1、联合体的大小为最大成员的大小,在联合体union Test中unsigned int是最大的占4个字节。

2、联合体共用一块内存,其内存大小为最大成员的内存大小,所以所以成员的地址都一样,&test = &test.a = &test.b = &test.c。

3、给联合体某个成员赋值时会影响到另外一个成员的数值,如下图所示:

如果是小端模式:

变量高位是放高地址、变量低位是放低地址。这里test.c=0x11223344指11为高位、44为低位。

栈区遵循“先进后出、后进先出”的规则,即打印的时候从高地址到地址依次打印,所以test.c打印值为11223344。

从上图可以看出,test.a、test.b和test.c的起使地址都是44对应的地址,所以他们的地址都是一样的。

如果test.a = 0x88,那么test.a、test.b和test.c本来的值都会发生改变,即test.a = 88,test.b = 3388,test.c = 11223388。

二、利用联合体处理浮点型数据

经过上面对联合体的简单介绍,想必不少同学已经知道接下来要说什么了吧。

直接上测试代码:

测试结果是什么呢?

输入的浮点数是6.91,我们可以看到uint8_data也有了4个值,从联合体的定义来看,这两个值应该是一样的,我们来验证下是不是如此:

浮点转16进制还是挺麻烦的,有精力的同学可以自己算算,没精力的同学可以使用这个链接的在线转换工具:

http://xnkiot.com/#/floating

转换之后的结果,跟我们上面代码运行的结果是一致的。

那么,这个转换有什么意义呢?
我们经常通讯串数过程中,比如串口传输的事单字节数据,对于float类型的数据就不能直接传输了,需要进行转化,使用联合体就可以很方便的进行“自动转换”了。

END

来源:小飞哥玩嵌入式

版权归原作者所有,如有侵权,请联系删除。

推荐阅读
你管这破玩意叫CPU?
RTOS实现双核MCU消息通信
入职Linux驱动工程师后,我才知道的真相…

→点关注,不迷路←

文章引用微信公众号"嵌入式微处理器",如有侵权,请联系管理员删除!

博客评论
还没有人评论,赶紧抢个沙发~
发表评论
说明:请文明发言,共建和谐网络,您的个人信息不会被公开显示。