【OpenSSL】OpenSSL实现Base64

Base 64概述和应用场景

概述

Base64就是将二进制数据转换为字符串的一种算法。

应用场景

  • 邮件编码
  • xml或则json存储二进制内容
  • 网页传递数据URL
  • 数据库中以文本形式存放二进制数据
  • 可打印的比特币钱包地址base58Check(hash校验)
  • 网页上可以将图片直接使用Base64表达
  • 公私密钥的文本文件

Base16(16进制)

Base164位, 一个Unicode字符编码需要8位 ,那就需要将一个字符分解成2部分。编码字节的值,对应Base64的值如下对照表:

字节值Base64编码
00
11
22
33
44
55
66
77
88
99
10A
11B
12C
13D
14E
15F

从零开始实现Base16编解码

代码如下:

#include <iostream>
using namespace std;static const char BASE16_ENC_TAB[] = "0123456789ABCDEF";
// '0'~'9'  => 48~57, 'A'~'E' => 65~70static const char BASE16_DEC_TAB[128] =
{-1, // 0-1, -1, -1, -1, -1,-1, -1, -1, -1, -1, // 1-10-1, -1, -1, -1, -1,-1, -1, -1, -1, -1, // 11-20-1, -1, -1, -1, -1,-1, -1, -1, -1, -1, // 21-30-1, -1, -1, -1, -1,-1, -1, -1, -1, -1, // 31-40-1, -1, -1, -1, -1,-1, -1, 0, 1, 2, // 41-503, 4, 5, 6, 7, 8, 9, -1, -1, -1, // 51-60-1, -1, -1, -1, 10, 11, 12, 13, 14, 15, // 61-70 'A'~'F'
};int Base16Encode(const unsigned char* in, int size, char* out)
{for (int i = 0; i < size; i++) {//1 一个字节取出高4位和低4位char h = (in[i] >> 4) ;  // 以为char l = (in[i] & 0x0f); // 0000 1111 //去掉高4位// 0~15映射到对应的字符out[i * 2] = BASE16_ENC_TAB[h];out[i * 2 + 1] = BASE16_ENC_TAB[l];}// base 16转码后空间扩大一倍 4位转成一个字符 1字节转成两个字符return size * 2;
}/**
* 将Base16字符转换成常规字符串
*/
int Base16Decode(const string &in, unsigned char* out) 
{// 将两个字符拼成一个字节for (int i = 0; i < in.size(); i+=2) {unsigned char ch = in[i]; //高位转换的字符unsigned char lh = in[i + 1]; // 低位转换的字符// 上面拿到的还是个字符, 要转换成原始的数据unsigned char h = BASE16_DEC_TAB[ch];unsigned char l = BASE16_DEC_TAB[lh];//out[i/2] = (h <<4) + l;out[i / 2] = h << 4 | l;}return in.size() / 2;
}int main()
{cout << "Test Base16" << endl;const unsigned char data[] = "测试Base16";int len = sizeof(data);char out1[1024] = { 0 };int res = Base16Encode(data, len, out1);cout << res << ":" << out1 << endl;string code(out1);unsigned char out2[1024] = { 0 };res = Base16Decode(code, out2);cout << res << ":" << out2 << endl;return 0;
}

Base64(64进制)

首先查看Base64的值码对应表

字节值Base64编码字节值Base64编码字节值Base64编码字节值Base64编码
0A16Q32g48w
1B17R33h49x
2C18S34i50y
3D19T35j51z
4E20U36k520
5F21V37l531
6G22W38m542
7H23X39n553
8I24Y40o564
9J25Z41p575
10K26a42q586
11L27b43t597
12M28c44s608
13N29d45t619
14O30e46u62+
15P31f47v63/

Base64编码要求把3个8位字节(3*8=24), 之后在6位的前面补两个0, 形成8位一个字节的形式。如果剩下的字符不足3个字节, 则用0填充,输出字符使用=,因此编码后输出的文本末尾可能会出现1或2个=

Open SSL BIO接口

  • BIO包含了多种接口,用于控制在BIO_METHOD中不同实现函数, 包括6种filter型和8种source/sink型应用场景。
  • BIO_new创建一个BIO对象.
  • 数据源:source/sink类型的BIO是数据源BIO_new(BIO_s_mem()),生存内存是数据源对象
  • 过滤:filter BIO就是把数据从一个BIO转换到另外一个BIO或应用接口 BIO_new(BIO_f_base64())
  • BIO链:一个BIO链通常包括一个source BIO和一个或多个filter BIO BIO_push(b64_bio, mem_bio);
  • 写编码, 读解码 BIO_write BIO_read_ex

Open SSL BIO实现Base64编解码

#include <iostream>
#include <openssl/bio.h>
#include <openssl/evp.h>
#include <openssl/buffer.h>
using namespace std;int Base64Encode(const unsigned char* in, int len, char* out_base64)
{if (!in || len < 0 || !out_base64) {return 0;}//创建内存源auto mem_bio = BIO_new(BIO_s_mem());if (!mem_bio) return 0;// base64 filterauto b64_bio = BIO_new(BIO_f_base64());//这个接口在头文件 evp.hif (!b64_bio) {BIO_free(mem_bio); //释放申请成功的空间return 0;}// 形成BIO链表//b64-mem// 形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果BIO_push(b64_bio, mem_bio);  // 2个链表(从链表头部,代表整个链表)//write 是编码 3字节 => 4字节 不足3字节补充0 和等于号int re = BIO_write(b64_bio, in, len); //将数据写入到链表头if (re < 0) {// 写入失败, 清空整个链表节点BIO_free_all(b64_bio);return 0;}// 刷新缓存, 写入链表的memBIO_flush(b64_bio);// 从链表源内存读取BUF_MEM* p_data = nullptr; // 需要引入BIO_get_mem_ptr(b64_bio, &p_data); //拿到编码数据了int out_size = 0;if (p_data) {memcpy(out_base64, p_data->data, p_data->length);out_size = p_data->length;}//执行完后, 清理空间BIO_free_all(b64_bio);return out_size;
}int Base64Decode(const char* in, int len, unsigned char* out_data) 
{if (!in || len <= 0 || !out_data){return 0;}// 内存源auto mem_bio = BIO_new_mem_buf(in, len);if (!mem_bio) {return 0;}// base64 过滤器auto b64_bio = BIO_new(BIO_f_base64());if (!b64_bio) {BIO_free(mem_bio);return 0;}//形成BIO链条BIO_push(b64_bio, mem_bio);//读取 解码 4字节转3字节size_t size = 0;BIO_read_ex(b64_bio, out_data, len, &size);BIO_free_all(b64_bio);return size;
}int main(int argc, char argv[])
{cout << "Test openssl BIO base64" << endl;unsigned char data[] = "测试Base64数据";int len = sizeof(data);char out[1024];int ret = Base64Encode(data, len, out);if (ret) {out[ret] = '\0';}cout << "base64:" << out << endl;unsigned char out_data[1024] = { 0 };//ret = Base64Decode(out, sizeof(out), out_data);// 这里不能用sizeof()  , 用计算字符长度ret = Base64Decode(out, strlen(out), out_data);cout << "encode :" << out_data << endl;
}

Open SSLBase64编码换行

Open SSLBase64在做编码操作的时候,默认情况下遇到64字节(不同平台不确定,)的时候就会进行换行操作。
例如将上面的输入内容改得过长.

unsigned char data[] = "测试Base64数据形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果";

执行的编码结果就会出现换行操作, 如下图所示:
在这里插入图片描述
只需要在写入待编码码内容前进行参数设置,就可以使其不换行

	//超过长度不还行BIO_set_flags(b64_bio, BIO_FLAGS_BASE64_NO_NL); 

编码方法的整体代码如下:


int Base64Encode(const unsigned char* in, int len, char* out_base64)
{if (!in || len < 0 || !out_base64) {return 0;}//创建内存源auto mem_bio = BIO_new(BIO_s_mem());if (!mem_bio) return 0;// base64 filterauto b64_bio = BIO_new(BIO_f_base64());//这个接口在头文件 evp.hif (!b64_bio) {BIO_free(mem_bio); //释放申请成功的空间return 0;}// 形成BIO链表//b64-mem// 形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果BIO_push(b64_bio, mem_bio);  // 2个链表(从链表头部,代表整个链表)//超过长度不还行BIO_set_flags(b64_bio, BIO_FLAGS_BASE64_NO_NL); //write 是编码 3字节 => 4字节 不足3字节补充0 和等于号//编码数据每64直接会加一个\n换行符号,并且结尾时也有换行符号int re = BIO_write(b64_bio, in, len); //将数据写入到链表头if (re < 0) {// 写入失败, 清空整个链表节点BIO_free_all(b64_bio);return 0;}// 刷新缓存, 写入链表的memBIO_flush(b64_bio);// 从链表源内存读取BUF_MEM* p_data = nullptr; // 需要引入BIO_get_mem_ptr(b64_bio, &p_data); //拿到编码数据了int out_size = 0;if (p_data) {memcpy(out_base64, p_data->data, p_data->length);out_size = p_data->length;}//执行完后, 清理空间BIO_free_all(b64_bio);return out_size;
}

执行结果,如下,编码的内容已经不会再换行了。
在这里插入图片描述
从上可以看出, 如果没有了末尾的换行符号。解码的时候又出现无法解码的问题, 因为解码是按照末尾的\n来执行结尾的。因为如下,如果我再编码内容的后面加上\n即可正确的进行解码操作了.

int main(int argc, char argv[])
{cout << "Test openssl BIO base64" << endl;unsigned char data[] = "测试Base64数据形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果形成链表, 往base64内存写数据, 进行编码,结果会传递到链表的下一个节点, 到mem中读取结果";int len = sizeof(data);char out[1024];int ret = Base64Encode(data, len, out);if (ret) {out[ret] = '\0';}cout << "base64:" << out << endl;// 手动加下行结尾的符号out[ret] = '\n';out[ret+1] = '\0';unsigned char out_data[1024] = { 0 };//ret = Base64Decode(out, sizeof(out), out_data);// 这里不能用sizeof()  , 用计算字符长度ret = Base64Decode(out, strlen(out), out_data);cout << "encode :" << out_data << endl;
}

另外一种解决办法就是直接也在解码代码中也加上对换行符号的忽略.


int Base64Decode(const char* in, int len, unsigned char* out_data) 
{if (!in || len <= 0 || !out_data){return 0;}// 内存源auto mem_bio = BIO_new_mem_buf(in, len);if (!mem_bio) {return 0;}// base64 过滤器auto b64_bio = BIO_new(BIO_f_base64());if (!b64_bio) {BIO_free(mem_bio);return 0;}//形成BIO链条BIO_push(b64_bio, mem_bio);//取消默认读取换行符号做结束的操作BIO_set_flags(b64_bio, BIO_FLAGS_BASE64_NO_NL);//读取 解码 4字节转3字节size_t size = 0;BIO_read_ex(b64_bio, out_data, len, &size);BIO_free_all(b64_bio);return size;
}

执行的结果都可以OK了。
在这里插入图片描述
综上问题, 编解码必须要一致。

Base58

由于Base64的方式存在一些问题,比如+,/在一些传输内容时容易出现特殊字符问题,还有0/O/o, l(小写母L)I(大写字母i)在某些情况下肉眼不易区分。Base58去掉了0(数字0)O(大写字母o)l(小写字母L)I(大写字母i),以及两个特殊字符+,/.一共去掉了6个字符,就剩下了58个字符。

字节值Base58编码字节值Base58编码字节值Base58编码字节值Base58编码
0116H32Z48q
1217J33a49r
2318K34b50s
3419L35c51t
4520M36d52u
5621N37e53v
6722P38f54w
7823Q39g55x
8924R40h56y
9A25S41i57z
10B26T42j
11C27U43k
12D28V44m
13E29W45n
14F30X46o
15G31Y47p

辗转相除法

  • 两个数的最大公约数等于它们中较小的数和两数只差的最大公约数
  • 欧几里德算法,是求最大公约数的算法
  • 两个数的最大公约数是指同时整除它们的最大正整数。辗转相除法的基本原理是两个数的最大公约数等于它们中较小的数和两数只差的最大公约数。
  • 如果要将1234转换成58进制;
  1. 1234 除以 58 ,商21, 余数为16,查表得H
  2. 用21除以58, 商0, 余数为21, 查表得N
  3. 如果待转的数前面又0直接附加编码1来代表,有多少个就附加多少个。

Base56输出字节数

  • 在编码后字符串中, 是从58个字符中选择,需要表示的位数是 l o g 2 58 log_{2}58 log258, 每一个字母代表的信息是 l o g 2 58 log_{2}58 log258.
  • 输入的字节: (length *8)位。
  • 预留的字符数量就是 ( l e n g t h ∗ 8 ) / l o g 2 58 (length*8)/log_{2}58 length8)/log258

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/87782.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机竞赛 深度学习交通车辆流量分析 - 目标检测与跟踪 - python opencv

文章目录 0 前言1 课题背景2 实现效果3 DeepSORT车辆跟踪3.1 Deep SORT多目标跟踪算法3.2 算法流程 4 YOLOV5算法4.1 网络架构图4.2 输入端4.3 基准网络4.4 Neck网络4.5 Head输出层 5 最后 0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; *…

【Vue.js】使用Element搭建首页导航左侧菜单

一&#xff0c;Mock.js 1.1 认识Mock.js Mock.js是一个用于前端开发中生成随机数据、模拟接口响应的 JavaScript 库。模拟数据的生成器&#xff0c;用来帮助前端调试开发、进行前后端的原型分离以及用来提高自动化测试效率 总结来说&#xff0c;Element中的Mock.js是一个用于…

怎样快速打开github.com

1访问这个网站很慢是因为有DNS污染&#xff0c;被一些别有用心的人搞了鬼了&#xff0c; 2还有一个重要原因是不同的DNS服务器解析的速度不一样。 1 建议设置dns地址为114.114.114.114.我觉得假设一个县城如果有一个DNS服务器的话&#xff0c;这个服务器很可能不会存储…

[论文笔记]P-tuning v2

引言 今天带来第五篇大模型微调论文笔记P-tuning v2: Prompt Tuning Can Be Comparable to Fine-tuning Across Scales and Tasks。 作者首先指出了prompt tuning的一些不足,比如在中等规模的模型上NLU任务表现不好,还不能处理困难的序列标记任务,缺乏统一应用的能力。 然…

【Spring Boot】实战:实现Session共享

🌿欢迎来到@衍生星球的CSDN博文🌿 🍁本文主要学习实现Session共享 🍁 🌱我是衍生星球,一个从事集成开发的打工人🌱 ⭐️喜欢的朋友可以关注一下🫰🫰🫰,下次更新不迷路⭐️💠作为一名热衷于分享知识的程序员,我乐于在CSDN上与广大开发者交流学习。 💠我…

wepack打包生产环境使用http-proxy-middleware做api代理转发的方法

首先安装http-proxy-middleware依赖&#xff0c;这个用npm和yarn安装都可以。 然后在express服务器的代码增加如下内容&#xff1a; const express require("express"); const app express(); const { createProxyMiddleware, fixRequestBody, } require("h…

华为云Stack的学习(八)

九、华为云Stack网络服务介绍 1.网络服务概览 1.1 租户界面的网络服务 租户登入ManageOne运营面后&#xff0c;可在服务列表中查看到网络服务。用户使用网络服务前管理员需要在Service OM上提前创建好外部网络。 1.2 华为云Stack网络服务全景图 1.3 网络服务承载网元 2.虚拟…

Tuxera NTFS 2022 for Mac破解版百度网盘免费下载安装激活教程

Mac打不开移动硬盘”有多种原因&#xff0c;解决办法也不尽相同。它可能是安装的NTFS for Mac读写软件版本和当前macOS系统的兼容问题、或者是Mac没有正常连接硬盘等等。本篇文章就将为您罗列出导致“Mac打不开移动硬盘”的原因和解决办法。 为此不得不使用著名的Tuxera NTFS …

最快的包管理器--pnpm创建vue项目完整步骤

1.用npm全局安装pnpm npm install -g pnpm 2.在要创建vue项目的包下进入cmd&#xff0c;输入&#xff1a; pnpm create vue 3.输入项目名字&#xff0c;选择Router,Pinia,ESLint,Prettier之后点确定 4.cd到创建好的项目 &#xff0c;安装依赖 cd .\刚创建好的项目名称\ p…

云原生Kubernetes:K8S配置资源管理

目录 一、理论 1.Secret 2.Secret创建 3.Secret使用 4.Configmap 5.Configmap创建 6.Configmap使用 二、实验 1.Secret创建 2.Secret使用 3.Configmap创建 4.Configmap使用 三、问题 1.变量引用生成资源报错 2.查看pod日志失败 3.创建configmap报错 4.YAML创建…

网络编程-UDP协议(发送数据和接收数据)

需要了解TCP协议的&#xff0c;可以看往期文章 https://blog.csdn.net/weixin_43860634/article/details/133274701 TCP/IP参考模型 通过此图&#xff0c;可以了解UDP所在哪一层级中 代码案例 发送数据 package com.hidata.devops.paas.udp;import java.io.IOException; …

ElementUI之登陆+注册->饿了吗完成用户登录界面搭建,axios之get请求,axios之post请求,跨域,注册界面

饿了吗完成用户注册登录界面搭建axios之get请求axios之post请求跨域注册界面 1.饿了吗完成用户注册登录界面搭建 将端口号8080改为8081 导入依赖&#xff0c;在项目根目录使用命令npm install element-ui -S&#xff0c;添加Element-UI模块 -g&#xff1a;将依赖下载node_glod…

配置OSPFv3引入外部路由及路由过滤 华为实验

1.1 实验介绍 1.1.1 关于本实验 在大型园区网络中&#xff0c;往往使用不同的路由协议进行组网&#xff0c;实现全网的网络互通。不同的协议间通信&#xff0c;除了路由协议本身&#xff0c;还需要引入外部路由及路由信息过滤等技术。 本章内容主要介绍OSPFv3路由过滤及引入外…

win使用git(保姆级教程)

序言 上学期间用的git并不多&#xff0c;但是从研三实习以及后面工作来看&#xff0c;git是一项必备技能&#xff0c;所以在此来学习一下。 下载git安装包 打开网站&#xff0c;根据需求来下载&#xff1b;一般按照如下方式进行下载&#xff1a; 然后安装的时候记得按下图勾…

论文笔记:ViTGAN: Training GANs with Vision Transformers

2021 1 intro 论文研究的问题是&#xff1a;ViT是否可以在不使用卷积或池化的情况下完成图像生成任务 即不用CNN&#xff0c;而使用ViT来完成图像生成任务将ViT架构集成到GAN中&#xff0c;发现现有的GAN正则化方法与self-attention机制的交互很差&#xff0c;导致训练过程中…

《优化接口设计的思路》系列:第四篇—接口的权限控制

系列文章导航 《优化接口设计的思路》系列&#xff1a;第一篇—接口参数的一些弯弯绕绕 《优化接口设计的思路》系列&#xff1a;第二篇—接口用户上下文的设计与实现 《优化接口设计的思路》系列&#xff1a;第三篇—留下用户调用接口的痕迹 《优化接口设计的思路》系列&#…

知识库搭建保姆级教程,如何从0到1完成知识库搭建

在这个信息爆炸的时代&#xff0c;如何获取、整理和应用知识成为了我们个体价值和企业核心竞争力打造的重要表现&#xff0c;搭建一个高效的知识库可以提升我们企业的竞争力&#xff0c;必要时还能快速切换赛道&#xff0c;开展一个新的领域。 今天我们将结合HelpLook 来与你一…

C++之互斥锁、读写锁、互斥量、 信号量、原子锁机制总结(二百二十五)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

MAC word 如何并列排列两张图片

系统&#xff1a;MAC os 参考博客 https://baijiahao.baidu.com/s?id1700824516945958911&wfrspider&forpc 步骤1 新建一个word文档和表格 修改表格属性 去掉自动重调尺寸以适应内容 插入图片 在表格的位置插入对应的图片如下 去除边框 最终结果如下

UE5 ChaosVehicles载具研究

一、基本组成 载具Actor类名称&#xff1a;WheeledVehiclePawn Actor最原始的结构 官方增加了两个摇臂相机&#xff0c;可以像驾驶游戏那样切换多机位、旋转观察 选择骨骼网格体、动画蓝图类、开启物理模拟 二、SportsCar_Pawn 角阻尼&#xff1a;物体旋转的阻力。数值越大…