关于自定义Base64编解码的实现

什么是Base64

Base64编码是将字符串以每3个8比特(bit)的字节子序列拆分成4个6比特(bit)的字节(6比特有效字节,最左边两个永远为0,其实也是8比特的字节)子序列,再将得到的子序列查找Base64的编码索引表,得到对应的字符拼接成新的字符串的一种编码方式。

每个3位8比特数据拆分成4个6比特数据过程如下图所示:

关于自定义Base64编解码的实现

 

 

 注意事项

Base64编码后的数据大小必须是4的倍数,当转换后的数据大小不是4的倍数时可以用‘=’号或者其他符号代替。

代码实现

以下是我自定义的Base64编解码实现规则

编码部分

const char base64_table[] = "在此添加自定义base64编码规则";
std::string encode(std::string &src)
{
    char encodeData[10000000];
// 以3个字节为一组
    int data_len = src.length() / 3;
    // std::cout << src.length() << std::endl;
    // src的余数数据
    int data_add = src.length() % 3;
    int temp = 0;
    int tmp = 0;
    int n = 0;
    while (temp < data_len)
    {
        encodeData[n++] = base64_table[src[tmp] >> 2];
        encodeData[n++] = base64_table[(src[tmp] & 0x3) << 4 | src[tmp + 1] >> 4];
        encodeData[n++] = base64_table[(src[tmp + 1] & 0x0f) << 2 | src[tmp + 2] >> 6];
        encodeData[n++] = base64_table[src[tmp + 2] & 0x3f];
        tmp += 3;
        temp++;
    }
    // 余数为1,需要补齐2个=
    if (data_add == 1)
    {
        encodeData[n++] = base64_table[src[tmp] >> 2];
        encodeData[n++] = base64_table[(src[tmp] & 0x3) << 4];
        encodeData[n++] = '=';
        encodeData[n++] = '=';
        encodeData[n] = '\0';
    }
    // 余数为2,需要补齐1个=
    else if (data_add == 2)
    {
        encodeData[n++] = base64_table[src[tmp] >> 2];
        encodeData[n++] = base64_table[(src[tmp] & 0x3) << 4 | src[tmp + 1] >> 4];
        encodeData[n++] = base64_table[(src[tmp + 1] & 0xf) << 2];
        encodeData[n++] = '=';
        encodeData[n] = '\0';
    }
    else
    {
        encodeData[n] = '\0';
    }
    return std::string(encodeData);
}

解码部分

std::string decode(std::string &src)
{
    // 自定义base64解码表,以ASCII码值顺序存储,对应base64编码表
    const char decode_base64_table[] = {“自定义128位base64解码规则,以1128位ASCII码的顺序存储base64编码表中对应字符的位置”};
    int inputLen = src.length();
    // 计算解码后的数据长度
    int outputLen = inputLen / 4 * 3;
    unsigned buf = 0;
    unsigned bufSize = 0;
    if (inputLen % 4 == 0)
    {
        if (src[inputLen - 1] == '=')
            outputLen--;
        if (src[inputLen - 2] == '=')
            outputLen--;
    }
    else
    {
        switch (inputLen % 4)
        {
        case 1:
            throw std::invalid_argument("invalid input-1");
            return nullptr;
        case 2:
            outputLen++;
            break;
        case 3:
            outputLen += 2;
            break;
        }
    }
    std::string output(outputLen, '\0');
    int i = 0;
    for (uint8_t c : src)
    {
        if (c > 127 || c == '=')
            break;
        uint8_t sextet = decode_base64_table[c];
        if (sextet == 64)
        {
            throw std::invalid_argument("invalid base64 input-2");
            return nullptr;
        }
        buf = (buf << 6) + sextet;
        bufSize += 6;
        if (bufSize >= 8)
        {
            output[i++] = (buf >> (bufSize - 8));
            bufSize -= 8;
        }
    }
    return output;
}

 

原文链接:https://www.cnblogs.com/a1557944/p/17266792.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:关于自定义Base64编解码的实现 - Python技术站

(0)
上一篇 2023年4月18日
下一篇 2023年4月18日

相关文章

  • 高效c语言1快速入门

    本章将开发你的第一个C语言程序:传统的 “Hello, world!”程序。然后讨论一些编辑器和编译器的选项,并阐述移植性问题。 Hello, world! #include <stdio.h> #include <stdlib.h> int main(void) { puts(“Hello, world!”); return EXI…

    C++ 2023年5月10日
    00
  • C++基础1: 命名空间

    0. 前言 C++是在C语言基础之上的一门语言, 所以学习C++的第一步是思考C++是如何优化C语言设计中的一些不足, 如作用域, IO , 函数, 指针, 宏等 这些内容同时也是为之后学习C++类和对象打基础, 下面说一下C++是如何优化C语言中的作用域 1. 命名空间 在说明命名空间的概念和如何优化C作用域之前, 首先来看一个例子 如图, 编译器报ran…

    C++ 2023年4月19日
    00
  • 网络框架重构之路plain2.0(c++23 without module) 环境

    接下来本来就直接打算分享框架重构的具体环节,但重构的代码其实并没有完成太多,许多的实现细节在我心中还没有形成一个定型。由于最近回归岗位后,新的开发环境需要自己搭建,搭建的时间来说花了我整整一天的时间才勉强搞定。人们常说工欲善其事必先利其器,开发环境和工具是必不可少的,否则你会发现在接下来的过程中遇到困难的时候就会走很多弯路。虽然最后我们仍旧达到了目的,但是我…

    C++ 2023年4月17日
    00
  • XMake学习笔记(1):Windows(MSYS2)下MinGW-w64环境搭建和XMake安装

    以前写的C++基本都是C with STL,大多是面向过程的算法题,或者比较小的项目,然后经常报各种编译错误(对编译原理不熟),经常把人搞到崩溃,搞不懂构建、链接之类的东西。 现在开始记录一下XMake的学习笔记,记录一些学习过程中踩的坑,在这篇文章,你将学习到Windows下利用MSYS2进行Mingw-w64环境搭建和XMake安装,并用Xmake构建一…

    C++ 2023年4月30日
    00
  • 如何将 Spire.Doc for C++ 集成到 C++ 程序中

    Spire.Doc for C++ 是一个专业的 Word 库,供开发人员在任何类型的 C++ 应用程序中阅读、创建、编辑、比较和转换 Word 文档。 本文演示了如何以两种不同的方式将 Spire.Doc for C++ 集成到您的 C++ 应用程序中。 通过 NuGet 安装 Spire.Doc for C++ 通过手动导入库安装 Spire.Doc f…

    C++ 2023年4月27日
    00
  • 【Visual Leak Detector】配置项 TraceInternalFrames

    说明 使用 VLD 内存泄漏检测工具辅助开发时整理的学习笔记。本篇介绍 VLD 配置文件中配置项 TraceInternalFrames 的使用方法。同系列文章目录可见 《内存泄漏检测工具》目录 目录 说明 1. 配置文件使用说明 2. 设置是否跟踪内部堆栈的调用 2.1 测试代码 2.2 TraceInternalFrames = no 时的输出 2.3 …

    C++ 2023年4月18日
    00
  • luogu_P1040 [NOIP2003 提高组] 加分二叉树

    P1040 [NOIP2003 提高组] 加分二叉树 – 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 题意:给你一颗中序遍历为1到n的二叉树,和每个节点的val。树的值=左子树的值×右子树的值+根的val,空树值为1,求整个树最大值和这个值树的前序遍历。 题解:区间dp。dp[l][r]表示最大值,root[l][r]表示最大值的根,枚举区…

    C++ 2023年4月27日
    00
  • 【Visual Leak Detector】配置项 StartDisabled

    说明 使用 VLD 内存泄漏检测工具辅助开发时整理的学习笔记。本篇介绍 VLD 配置文件中配置项 StartDisabled 的使用方法。同系列文章目录可见 《内存泄漏检测工具》目录 目录 说明 1. 配置文件使用说明 2. 设置是否禁用自动初始化 2.1 测试代码 2.2 StartDisabled = no 时的输出 2.3 StartDisabled …

    C++ 2023年4月18日
    00
合作推广
合作推广
分享本页
返回顶部