【转】玩转嵌入式(公众号)
在入门单片机时,想必大家都都会遇到一下这种情况
unsigned char a = 0x12;
unsigned char b = 0x34;
unsigned int c = 0;
如何把两个8位数据和在一起变成16位数据呢?
一般情况下大家都会这样做,我最初是也是这么做的
方法1 【使用移位指令】
int c = (a<<8)|b;
方法2 【使用指针】
unsigned char *cptr;
cptr = (unsigned char*)(&d);
cptr[0] = a;
cptr[1] = b;
方法3 【强制指针类型转换】
*((unsigned char*)(&d)) = a;
*((unsigned char*)(&d)+1) = b;
或
((unsigned char*)(&d))[0] = a;
((unsigned char*)(&d))[1] = b;
以上这三种方法都是没有错误的,但在keil编译器中编译出的结果是不一样的。第三种方法编译出的代码会更简洁
今天就交给大家第4种方法
方法4 【联合体】
typedef union{
unsigned int i;
unsigned char c[2];
}u_int;
unsigned char dH = 0x11, dL=0x22;
unsigned int d;
u_int ud;
ud.c[0] = dH;
ud.c[1] = dL;
d = ud.i;
此时d = 0x1122;
这里就是利用了联合体union的特性来实现把两个8位数据合并成一个16位数据的方法。在C语言里操作指针最容易出现错误,所以在遇到这样类似的问题大家不妨使用联合体的方式进行处理数据,既不容易出现错误,生成的代码又简洁。
测试代码
#include "stdio.h" typedef unsigned char uint8_t;
typedef unsigned short uint16_t;
typedef unsigned int uint32_t; uint8_t a = 0x12;
uint8_t b = 0x34;
uint16_t c; /* 使用移位转换类型 */
void use_shift(uint16_t *val)
{
*val = (a<<) | b;
} /* 使用指针转换类型 */
void use_point(uint16_t *val)
{
uint8_t *p; p = (uint8_t *)val;
p[] = b;
p[] = a;
} /* 强制指针转换类型 */
void force_point_convert(uint16_t *val)
{
#if 0
*((uint8_t *)val) = b;
*((uint8_t *)val + ) = a;
#else
((uint8_t *)val)[] = b;
((uint8_t *)val)[] = a;
#endif
} /* 使用共用体转换类型 */
typedef union
{
uint16_t int16;
uint8_t char8[];
} CVR_u;
CVR_u Obj; void use_union(CVR_u *p)
{
p->char8[] = b;
p->char8[] = a;
} int main(void)
{
c = ;
use_shift(&c);
printf("c1= %#X\n", c); c = ;
use_point(&c);
printf("c2= %#X\n", c); c = ;
force_point_convert(&c);
printf("c3= %#X\n", c); c = ;
use_union(&Obj);
c = Obj.int16;
printf("c4= %#X\n", c); return ;
}