嵌入式大杂烩周记 | 第 14 期

举报
嵌入式大杂烩 发表于 2022/08/19 23:03:36 2022/08/19
【摘要】 大家好,我是杂烩君。 嵌入式大杂烩周记主要是一些实用项目学习分享,每篇一个主题。 内容主要来源于我们之前收集的资料: https://gitee.com/zhengnianli/EmbedSummary 本期主角:base64 在我们嵌入式开发中,我们常常都会用到一些通用的工具库,往往都会有个base64编解码...

大家好,我是杂烩君。

嵌入式大杂烩周记主要是一些实用项目学习分享,每篇一个主题。

内容主要来源于我们之前收集的资料:

https://gitee.com/zhengnianli/EmbedSummary

本期主角:base64

在我们嵌入式开发中,我们常常都会用到一些通用的工具库,往往都会有个base64编解码模块。

base64编解码模块有什么用?

Base64就是一种基于64个可打印字符来表示二进制数据的方法,网络上最常见的用于传输8Bit字节码的编码方式之一。

比如,如果一个传输协议是基于ASCII文本的,那么它就不能传输二进制流,那你要将二进制流传输就得编码,因为有些8Bit字节码并没有对应的ASCII字符。

比如,我之前也没了解过base64。某个项目中在调试设备、云端、手机APP之前的通信时,设备端给手机端传一些8Bit字节码数据,手机端并未解析得到正确的数据,后来查了资料才知道需要进行base64编码,需要使用base64来屏蔽传输上的差异。后来,给数据套了一层base64之后,就正常了。

实际中,设备、云端、手机APP交互数据常常这么做:

设备端把base64编码后的数据封装在json字符串里,手机端先解析json拿到value,再进行base64解码拿到想要的数据。

注意:虽然编码之后的数据与加密一样都具有不可见性,但编码与加密的概念并不一样。编码是公开的,任何人都可以解码;而加密则相反,你只希望自己或者特定的人才可以对内容进行解密。

base64编解码库网络上有很多,这里分享一份:

https://blog.csdn.net/qq_15762939/article/details/110202212

base64编解码测试

1、base64基本原理

Base64是基于64个可打印字符来表示二进制数据的编解码方式,每个字符对应一个索引,对应关系表如:

8943396d3548046209bc8e802873539b.jpeg

一个Base64字符实际上代表着6个二进制位(bit),4个Base64字符对应3字节字符串/二进制数据。

3个字符为一组的的base64编码方式如:

6ba6a340855ee5499ed2a78a0290ee6b.png

小于3个字符为一组的编码方式如:

07405afd6ac9d8b103b12136aff67c64.png

2、base64测试

我们对如下三种数据进行编解码测试:

  • "hello world abcdefg"

  • "hello ,,,,,,,,, world"

  • {0x81, 0x82, 0x83, 0x84}

测试代码:

// 微信公众号:嵌入式大杂烩
#include <stdio.h>
#include <string.h>
#include <math.h>

/* Base64编码映射表 */
const char *const Base64Table = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

int Base64Encode(const unsigned char *Bindata, int Binlen, char *const Base64Buf);
int Base64Decode(const char *Base64Buf, unsigned char *const Bindata);

int main(int argc, char **argv)
{
    unsigned char base64_buf[128] = {0};
    unsigned char bin_buf[128] = {0};
    int base64_len = 0;
    int bin_len = 0;

    printf("\n=================================test1==========================================\n");
    char *test_data1 = "hello world abcdefg";

    printf("src data str = %s, src_data_len = %ld\n", test_data1, strlen(test_data1));
    memset(base64_buf, 0x00, sizeof(base64_buf));
    memset(bin_buf, 0x00, sizeof(bin_buf));
    base64_len = Base64Encode(test_data1, strlen(test_data1), base64_buf);
    printf("after base64_encode, base64_buf = %s, base64_len = %d\n", base64_buf, base64_len);
    bin_len = Base64Decode(base64_buf, bin_buf);
    printf("after base64_decode bin_buf = %s, bin_len = %d\r\n", bin_buf, bin_len);

    printf("\n=================================test2==========================================\n");
    char *test_data2 = "hello ,,,,,,,,, world";

    printf("src data str = %s, src_data_len = %ld\n", test_data2, strlen(test_data2));
    memset(base64_buf, 0x00, sizeof(base64_buf));
    memset(bin_buf, 0x00, sizeof(bin_buf));
    base64_len = Base64Encode(test_data2, strlen(test_data2), base64_buf);
    printf("after base64_encode, base64_buf = %s, base64_len = %d\n", base64_buf, base64_len);
    bin_len = Base64Decode(base64_buf, bin_buf);
    printf("after base64_decode bin_buf = %s, bin_len = %d\r\n", bin_buf, bin_len);

    printf("\n=================================test3==========================================\n");
    unsigned char test_data3[4] = {0x81, 0x82, 0x83, 0x84};

    printf("src data hex = ");
    for (int i = 0; i < sizeof(test_data3); i++)
    {
        printf("%#02x ", test_data3[i]);
    }
    printf("   src_data_len = %ld\n", sizeof(test_data3));
    memset(base64_buf, 0x00, sizeof(base64_buf));
    memset(bin_buf, 0x00, sizeof(bin_buf));
    base64_len = Base64Encode(test_data3, sizeof(test_data3), base64_buf);
    printf("after base64_encode, base64_buf = %s, base64_len = %d\n", base64_buf, base64_len);
    bin_len = Base64Decode(base64_buf, bin_buf);

    printf("after base64_decode, data hex = ");
    for (int i = 0; i < bin_len; i++)
    {
        printf("%#02x ", bin_buf[i]);
    }
    printf("   bin_len = %d\n", bin_len);
    printf("\n");

    return 0;
}

/******************************************************** 
 *功能描述:Base64编码
 *输入参数: Bindata:原始bin数据
    Binlen:原始 bin数据长度
 *输出参数:Base64Buf:base64编码数据
 *返 回 值:base64编码后数据长度
*********************************************************/
int Base64Encode(const unsigned char *Bindata, int Binlen, char *const Base64Buf)
{
    unsigned char s8CharIndex = 0;
    int i=0, Len=0;

    for ((i=0,Len=0); i<Binlen; i+=3)
    {
        s8CharIndex = (Bindata[i]>>2);
        s8CharIndex &= (unsigned char)0x3F;
        Base64Buf[Len++] = Base64Table[(int)s8CharIndex];
        s8CharIndex = ((unsigned char)(Bindata[i]<<4)) & ((unsigned char)0x30);
        if ((i+1) >= Binlen)
        {
            Base64Buf[Len++] = Base64Table[(int)s8CharIndex];
            Base64Buf[Len++] = '=';
            Base64Buf[Len++] = '=';
            break;
        }

        s8CharIndex |= ((unsigned char)(Bindata[i+1]>>4)) & ((unsigned char)0x0F);
        Base64Buf[Len++] = Base64Table[(int)s8CharIndex];
        s8CharIndex = ((unsigned char)(Bindata[i+1]<<2)) & ((unsigned char)0x3C);
        if ((i+2) >= Binlen)
        {
            Base64Buf[Len++] = Base64Table[(int)s8CharIndex];
            Base64Buf[Len++] = '=';
            break;
        }

        s8CharIndex |= ((unsigned char)(Bindata[i+2]>>6) & ((unsigned char)0x03));
        Base64Buf[Len++] = Base64Table[(int)s8CharIndex];
        s8CharIndex = ((unsigned char)Bindata[i+2]) & ((unsigned char)0x3F) ;
        Base64Buf[Len++] = Base64Table[(int)s8CharIndex];
    }

    return Len;
}  

/******************************************************** 
 *功能描述:Base64解码
 *输入参数: Base64Buf:base64编码数据
 *输出参数:Bindata:解码后bin数据
 *返 回 值:解码后bin数据长度
*********************************************************/
int Base64Decode(const char *Base64Buf, unsigned char *const Bindata)
{
    int i, Len=0;
    unsigned char s8CharIndex = 0;
    unsigned char temp[4] = {0};
    for ((i=0,Len=0); Base64Buf[i]!='\0'; i+=4)
    {
        memset(temp, 0xFF, sizeof(temp));
        for (s8CharIndex=0; s8CharIndex<64; s8CharIndex++)
        {
            if (Base64Table[s8CharIndex] == Base64Buf[i])
                temp[0]= s8CharIndex;
        }

        for (s8CharIndex=0; s8CharIndex<64; s8CharIndex++)
        {
            if (Base64Table[s8CharIndex] == Base64Buf[i+1])
                temp[1]= s8CharIndex;
        }

        for (s8CharIndex=0; s8CharIndex<64; s8CharIndex++)
        {
            if (Base64Table[s8CharIndex] == Base64Buf[i+2])
                temp[2]= s8CharIndex;
        }

        for (s8CharIndex=0; s8CharIndex<64; s8CharIndex++)
        {
            if (Base64Table[s8CharIndex] == Base64Buf[i+3])
                temp[3]= s8CharIndex;
        }

        if ((0xFF==temp[0]) || (0xFF==temp[1]) || (0xFF==temp[2]) || (0xFF==temp[3]))
        {
            //printf("(%s:%d) already decode base64 Len:%d\r\n", __func__, __LINE__, i);
            //break;//考虑到有些base64是经过变异的,不做退出处理
        }

        Bindata[Len++] = ((unsigned char)(((unsigned char)(temp[0] << 2))&0xFC)) |
        ((unsigned char)((unsigned char)(temp[1]>>4)&0x03));
        if (Base64Buf[i+2] == '=')
        {
            printf("(%s:%d) already decode base64 Len:%d\r\n", __func__, __LINE__, i);
            break;
        }

        Bindata[Len++] = ((unsigned char)(((unsigned char)(temp[1] << 4))&0xF0)) |
        ((unsigned char)((unsigned char)(temp[2]>>2)&0x0F));
        if (Base64Buf[i+3] == '=')
        {
            printf("(%s:%d) already decode base64 Len:%d\r\n", __func__, __LINE__, i);
            break;
        }

        Bindata[Len++] = ((unsigned char)(((unsigned char)(temp[2] << 6))&0xF0)) |
        ((unsigned char)(temp[3]&0x3F));
    }

    return Len;
}

运行:

532ff9c02d6fbc4aa0be782d04b80285.png

以上就是本次的分享。

如果觉得文章有帮助,麻烦帮忙点赞、收藏、转发,谢谢!

咱们下期见~

参考:

https://blog.csdn.net/qq_15762939/article/details/110202212
https://www.zhihu.com/question/36306744
https://baijiahao.baidu.com/s?id=1735577033729027737&wfr=spider&for=pc

注意

由于微信公众号近期改变了推送规则,为了防止找不到,可以星标置顶,这样每次推送的文章才会出现在您的订阅列表里。

猜你喜欢:

分享几个实用的代码片段(第二弹)

分享一种你可能不知道的bug定位方法

分享一种修改配置文件的方法

《嵌入式大杂烩周记第 13 期:lz4》

《嵌入式并行多线程处理器,了解一下!》

《分享一种修改配置文件的方法》

《分享几个实用的代码片段(附代码例子)》

《废旧板子再利用:搭建无线调试环境!》

《嵌入式段错误的3种调试方法汇总!》

《简说TCP通信非阻塞接收(附代码例子)》

《TCP server如何与多个client通信?》

《TCP通信常用接口的使用封装》

《写国际化的嵌入式代码,时间问题如何处理?》

《Linux命令行万能解压命令》

《嵌入式软件中,总线错误的坑?替大家先踩一步》

《分享嵌入式软件调试方法及几个有用的工具!》

《分享两点提高编程能力的建议!》

在公众号聊天界面回复1024,可获取嵌入式资源;回复 m ,可查看文章汇总

文章来源: blog.csdn.net,作者:嵌入式大杂烩,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/zhengnianli/article/details/126397395

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。