嵌入式编程中的高低位交换如何实现?

嵌入式ARM 2020-12-01 00:00

最近的协议中遇到字节高低位转换的问题,于是偷懒上网查看,遇到类似的问题,也认识一个新的名字,叫做蝶式交换

问题是这样子的

协议要求字节低位在左,高位在右,对每个字节做转换处理,逐个交换其高低位,例如11010001,经过0->7,1->6,2->5,3->4对应位的交换,变成10001011

这里就需要有位操作的知识,参考 位操作运算有什么奇技淫巧?(附源码)

这里我们以8位数据高低位转换为例子:

 1#include <stdio.h>
2
3unsigned char highAndLowShiftHEX(unsigned char data);
4void printBin(int n);
5
6int main () {
7
8    highAndLowShiftHEX(209);
9    return 0;
10}
11
12unsigned char highAndLowShiftHEX(unsigned char data)  
13
{  
14   unsigned char i;  
15   unsigned char tmp=0x00;  
16   for(i=0;i<8;i++)  
17   {  
18       tmp=((data>>i)&0x01)|tmp;  
19       //printBin(tmp);
20       if(i<7)  
21           tmp=tmp<<1;  
22   }  
23   printf("\nafter shift data:");  
24   printBin(tmp);
25   return tmp;    
26}
27
28//由于二进制直观,故写了一个打印二进制的函数
29void printBin(int n)
30
{
31    int len = sizeof(n)*8;//总位数。
32    int i;
33    if(i == 0)
34    {
35         printf("0");
36         return;
37    }
38    for(i = len-1; i >= 0; i --)//略去高位0.
39    {
40        if(n&(1<<i)) break;
41    }
42
43    for(;i>=0; i --)
44        printf("%d", (n&(1<<i)) != 0);
45}

熟悉位操作,上述代码实现起来就比较简单。在嵌入式开发中这类问题通常使用蝶式交换法和查表法来实现。

高阶实现?

查表法就是将一些值存到内存中,在需要计算的时候查表,但是这种方法会占用额外的存储空间。

所以这里主要介绍一下蝶式交换法,我们以8位数据转换为例。

  • 假设原始序列为:1 2 3 4 5 6 7 8

  • 目标序列为:8 7 6 5 4 3 2 1

那么流程图如下:

由此完成了整个位的逆序转换,同样以11010001为例,下面是具体的实现代码:

 1#include <stdio.h>
2
3unsigned char highAndLowShiftHEX(unsigned char );
4void printBin(int );
5
6int main () {
7
8    highAndLowShiftHEX(209);
9    return 0;
10}
11
12unsigned char highAndLowShiftHEX(unsigned char data)  
13
{  
14   data=(data<<4)|(data>>4);  
15   data=((data<<2)&0xcc)|((data>>2)&0x33);  
16   data=((data<<1)&0xaa)|((data>>1)&0x55);  
17   printf("  after shift data=%x \n",data);  
18   printBin(data);
19   return data;  
20}
21
22//由于二进制直观,故写了一个打印二进制的函数
23void printBin(int n)
24
{
25    int len = sizeof(n)*8;//总位数。
26    int i;
27    if(i == 0)
28    {
29         printf("0");
30         return;
31    }
32    for(i = len-1; i >= 0; i --)//略去高位0.
33    {
34        if(n&(1<<i)) break;
35    }
36
37    for(;i>=0; i --)
38        printf("%d", (n&(1<<i)) != 0);
39}

交换字节的高低位并不是一个很常见的问题,遇到该问题时,需要经过仔细的分析,加上对C语言位操作的熟练掌握,就能够很好的解决这一类的问题。

扩展

那么我们将其扩展到16位半字数据的高低位转换。

其实原理是跟8位的时候是一样的,采用简单的移位方式对数据的高低位进行转换。熟悉位操作的话代码就相对简单了。

下面是对该思路的具体实现。

 1#include <stdio.h>
2
3void expandPrintBin(int val2);
4unsigned short HighAndLowSwitchHEX(unsigned short data);
5
6int main () {
7
8    HighAndLowSwitchHEX(38491);
9    return 0;
10}
11
12
13//由于二进制直观,故写了一个扩展的打印二进制的函数
14void expandPrintBin(int val2)
15
{
16    int i,k;
17    unsigned char *p = (unsigned char*)&val2 + 3//从低位到高位,低端字节计算机
18    for( k = 0; k <= 3; k++)
19    {
20        int val2 = *(p-k);
21        for (i = 7; i >= 0; i--)
22        {
23            if(val2 & (1 << i))
24                printf("1");
25            else
26                printf("0");
27        }
28        printf(" ");
29    }
30}
31unsigned short HighAndLowSwitchHEX(unsigned short data)
32
{
33    unsigned char i = 0;
34    unsigned short temp = 0x0000;
35
36    for(i = 0; i < 16; i++)
37    {
38        temp = ((data >> i) & 0x0001) | temp;
39        if(i < 15)
40        {
41            temp = temp << 1;
42        }
43    }
44    printf("temp:%x\n\n",temp);
45    expandPrintBin(temp);
46    return temp;
47}

同样的所谓的蝶式交换法,我引用了字节交换法的例子,我们可以演算一下:

  • 假设原始序列为:a b c d e f g h i j k l m n o p

  • 目标序列为:p o n m l k j i h g f e d c b a

那么流程图如下:

由此完成了整个位的逆序转换,完成了算法的拓展,以1001011001011011为例,下面是具体的实现代码:

 1#include <stdio.h>
2
3unsigned short highAndLowShiftHEX(unsigned short data);
4void expandPrintBin(int val2);
5
6int main () {
7
8    highAndLowShiftHEX(38491);
9    return 0;
10}
11
12unsigned short highAndLowShiftHEX(unsigned short data)
13
{
14  data = (data << 8) | (data >> 8);   //0101101110010110
15  data = ((data << 4) & 0xF0FF) | ((data >> 4) & 0xFF0F); //1011010101101001
16  data = ((data << 2) & 0xCCCC) | ((data >> 2) & 0x3333); //1110010110010110
17  data = ((data << 1) & 0xAAAA) | ((data >>1 ) & 0x5555); //1101101001101001
18  printf("  after shift data=%x \n",data);  
19  expandPrintBin(data);
20  return data;  
21}
22
23//由于二进制直观,故写了一个扩展的打印二进制的函数
24void expandPrintBin(int val2)
25
{
26    int i,k;
27    unsigned char *p = (unsigned char*)&val2 + 3//从低位到高位,低端字节计算机
28    for( k = 0; k <= 3; k++)
29    {
30        int val2 = *(p-k);
31        for (i = 7; i >= 0; i--)
32        {
33            if(val2 & (1 << i))
34                printf("1");
35            else
36                printf("0");
37        }
38        printf(" ");
39    }
40}

对于这种交换位的问题,在嵌入式开发中会经常遇到,熟悉掌握位操作还有调试手段很重要。熟练的利用好C语言的移位操作,就能快速的解决此类问题。

总结

上面的问题关键还是位操作的灵活妙用,其次是写了两个打印二进制的函数,方便使用,代码可以直接使用,调试不易,欢迎点赞,在看,转发也是厚爱,我们下一期,再见!

END

来源:技术让梦想更伟大,作者:李肖遥

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

推荐阅读

飞机上一般是什么操作系统?

高速CAN、容错CAN、LIN总线有什么区别?

大佬终于把鸿蒙OS讲明白了,收藏了!

嵌入式ARM 关注这个时代最火的嵌入式ARM,你想知道的都在这里。
评论
  • 当前,智能汽车产业迎来重大变局,随着人工智能、5G、大数据等新一代信息技术的迅猛发展,智能网联汽车正呈现强劲发展势头。11月26日,在2024紫光展锐全球合作伙伴大会汽车电子生态论坛上,紫光展锐与上汽海外出行联合发布搭载紫光展锐A7870的上汽海外MG量产车型,并发布A7710系列UWB数字钥匙解决方案平台,可应用于数字钥匙、活体检测、脚踢雷达、自动泊车等多种智能汽车场景。 联合发布量产车型,推动汽车智能化出海紫光展锐与上汽海外出行达成战略合作,联合发布搭载紫光展锐A7870的量产车型
    紫光展锐 2024-12-03 11:38 126浏览
  • 概述 说明(三)探讨的是比较器一般带有滞回(Hysteresis)功能,为了解决输入信号转换速率不够的问题。前文还提到,即便使能滞回(Hysteresis)功能,还是无法解决SiPM读出测试系统需要解决的问题。本文在说明(三)的基础上,继续探讨为SiPM读出测试系统寻求合适的模拟脉冲检出方案。前四代SiPM使用的高速比较器指标缺陷 由于前端模拟信号属于典型的指数脉冲,所以下降沿转换速率(Slew Rate)过慢,导致比较器检出出现不必要的问题。尽管比较器可以使能滞回(Hysteresis)模块功
    coyoo 2024-12-03 12:20 170浏览
  • 11-29学习笔记11-29学习笔记习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记
    youyeye 2024-12-02 23:58 92浏览
  •         温度传感器的精度受哪些因素影响,要先看所用的温度传感器输出哪种信号,不同信号输出的温度传感器影响精度的因素也不同。        现在常用的温度传感器输出信号有以下几种:电阻信号、电流信号、电压信号、数字信号等。以输出电阻信号的温度传感器为例,还细分为正温度系数温度传感器和负温度系数温度传感器,常用的铂电阻PT100/1000温度传感器就是正温度系数,就是说随着温度的升高,输出的电阻值会增大。对于输出
    锦正茂科技 2024-12-03 11:50 141浏览
  • 光伏逆变器是一种高效的能量转换设备,它能够将光伏太阳能板(PV)产生的不稳定的直流电压转换成与市电频率同步的交流电。这种转换后的电能不仅可以回馈至商用输电网络,还能供独立电网系统使用。光伏逆变器在商业光伏储能电站和家庭独立储能系统等应用领域中得到了广泛的应用。光耦合器,以其高速信号传输、出色的共模抑制比以及单向信号传输和光电隔离的特性,在光伏逆变器中扮演着至关重要的角色。它确保了系统的安全隔离、干扰的有效隔离以及通信信号的精准传输。光耦合器的使用不仅提高了系统的稳定性和安全性,而且由于其低功耗的
    晶台光耦 2024-12-02 10:40 143浏览
  • 最近几年,新能源汽车愈发受到消费者的青睐,其销量也是一路走高。据中汽协公布的数据显示,2024年10月,新能源汽车产销分别完成146.3万辆和143万辆,同比分别增长48%和49.6%。而结合各家新能源车企所公布的销量数据来看,比亚迪再度夺得了销冠宝座,其10月新能源汽车销量达到了502657辆,同比增长66.53%。众所周知,比亚迪是新能源汽车领域的重要参与者,其一举一动向来为外界所关注。日前,比亚迪汽车旗下品牌方程豹汽车推出了新车方程豹豹8,该款车型一上市就迅速吸引了消费者的目光,成为SUV
    刘旷 2024-12-02 09:32 138浏览
  • 作为优秀工程师的你,已身经百战、阅板无数!请先醒醒,新的项目来了,这是一个既要、又要、还要的产品需求,ARM核心板中一个处理器怎么能实现这么丰富的外围接口?踌躇之际,你偶阅此文。于是,“潘多拉”的魔盒打开了!没错,USB资源就是你打开新世界得钥匙,它能做哪些扩展呢?1.1  USB扩网口通用ARM处理器大多带两路网口,如果项目中有多路网路接口的需求,一般会选择在主板外部加交换机/路由器。当然,出于成本考虑,也可以将Switch芯片集成到ARM核心板或底板上,如KSZ9897、
    万象奥科 2024-12-03 10:24 96浏览
  • RDDI-DAP错误通常与调试接口相关,特别是在使用CMSIS-DAP协议进行嵌入式系统开发时。以下是一些可能的原因和解决方法: 1. 硬件连接问题:     检查调试器(如ST-Link)与目标板之间的连接是否牢固。     确保所有必要的引脚都已正确连接,没有松动或短路。 2. 电源问题:     确保目标板和调试器都有足够的电源供应。     检查电源电压是否符合目标板的规格要求。 3. 固件问题: &n
    丙丁先生 2024-12-01 17:37 114浏览
  • TOF多区传感器: ND06   ND06是一款微型多区高集成度ToF测距传感器,其支持24个区域(6 x 4)同步测距,测距范围远达5m,具有测距范围广、精度高、测距稳定等特点。适用于投影仪的无感自动对焦和梯形校正、AIoT、手势识别、智能面板和智能灯具等多种场景。                 如果用ND06进行手势识别,只需要经过三个步骤: 第一步&
    esad0 2024-12-04 11:20 103浏览
  • 遇到部分串口工具不支持1500000波特率,这时候就需要进行修改,本文以触觉智能RK3562开发板修改系统波特率为115200为例,介绍瑞芯微方案主板Linux修改系统串口波特率教程。温馨提示:瑞芯微方案主板/开发板串口波特率只支持115200或1500000。修改Loader打印波特率查看对应芯片的MINIALL.ini确定要修改的bin文件#查看对应芯片的MINIALL.ini cat rkbin/RKBOOT/RK3562MINIALL.ini修改uart baudrate参数修改以下目
    Industio_触觉智能 2024-12-03 11:28 110浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦