openssl研发之base64编解码实例

一、base64编码介绍

Base64编码是一种将二进制数据转换成ASCII字符的编码方式。它主要用于在文本协议中传输二进制数据,例如电子邮件的附件、XML文档、JSON数据等。

Base64编码的特点如下:

  1. 字符集: Base64编码使用64个字符来表示二进制数据,通常使用字母(A-Z、a-z)、数字(0-9)和两个特殊字符(+ 和 /)。有时候,根据具体需求,可能会使用不同的字符集,比如URL安全的Base64编码会使用字符集 A-Za-z0-9-_,其中 “-” 和 “_” 替代了 “+” 和 “/”。

  2. 长度增加: Base64编码通常会导致编码后的数据长度略微增加。每三个字节的二进制数据编码成四个Base64字符。

  3. 填充字符: 如果最后一组不足三个字节,Base64编码会使用一个或两个等号 “=” 进行填充。

  4. 可逆性: Base64编码是可逆的,可以通过解码还原回原始的二进制数据。

  5. 广泛应用: Base64编码广泛应用于数据传输,特别是在文本协议中,以及在一些数据存储和表示的场景中。

下面是一个简单的示例,展示如何对字符串 “Hello, World!” 进行Base64编码:

原始字符串:Hello, World!

Base64编码:SGVsbG8sIFdvcmxkIQ==

二、base64编码原理

Base64编码的原理基于将二进制数据转换成一组可打印字符的过程。它是一种用64个字符来表示所有可能的6位二进制数的方法。以下是Base64编码的基本原理:

  1. 将数据划分成固定长度的块: 首先,将要编码的数据按照一定的规则划分成固定长度的块。通常是每三个字节划分为一组,因为每组有3个字节(24位),正好可以表示为4个Base64字符。

  2. 将块中的二进制数据转换为整数: 将每个块中的三个字节的二进制数据转换为一个24位的整数。这个整数的取值范围是0到2^24-1。

  3. 将整数按6位一组划分: 将上一步得到的整数按照6位一组划分成四组。因为2^6等于64,所以每组的取值范围是0到63。

  4. 将每组的值映射到Base64字符集: 将每一组得到的数值映射到Base64字符集。Base64字符集通常包含64个字符,例如字母A-Z、a-z、数字0-9,以及两个额外的字符(通常是"+“和”/")。这样就得到了四个Base64字符。

  5. 处理末尾的填充: 如果原始数据的长度不是3的倍数,那么最后一个块可能不足3个字节。在这种情况下,会使用一个或两个等号 “=” 进行填充,以保证每个块都是4个Base64字符。

举个例子,假设要编码的数据是"ABC",对应的ASCII码为65、66、67,转换成二进制分别是01000001、01000010、01000011。将这三个字节拼接在一起得到24位的二进制数,然后按照上述步骤转换为Base64编码。最后得到的Base64编码为"QUJD"。

这个过程是可逆的,也就是说,可以通过反向的过程将Base64编码解码回原始的二进制数据。在实际应用中,Base64编码常用于在文本协议中传输二进制数据,或者将二进制数据嵌入到文本文件中。

三、base64开发实例

#include <iostream>
#include <cstring>
#include <openssl/bio.h>
#include <openssl/evp.h>
#include <openssl/buffer.h>std::string base64_encode(const unsigned char* input, int length) {BIO* bio = BIO_new(BIO_f_base64());BIO_set_flags(bio, BIO_FLAGS_BASE64_NO_NL);BIO* bmem = BIO_new(BIO_s_mem());bio = BIO_push(bio, bmem);BIO_write(bio, input, length);BIO_flush(bio);BUF_MEM* bptr;BIO_get_mem_ptr(bio, &bptr);std::string result(bptr->data, bptr->length);BIO_free_all(bio);return result;
}std::string base64_decode(const char* input) {BIO* bio = BIO_new(BIO_f_base64());BIO_set_flags(bio, BIO_FLAGS_BASE64_NO_NL);BIO* bmem = BIO_new_mem_buf(input, -1);bio = BIO_push(bio, bmem);char buffer[1024];int length = BIO_read(bio, buffer, sizeof(buffer));BIO_free_all(bio);return std::string(buffer, length);
}int main() {const char* originalText = "Hello, Base64!";int originalLength = std::strlen(originalText);// Base64 编码std::string encodedText = base64_encode(reinterpret_cast<const unsigned char*>(originalText), originalLength);std::cout << "Base64 Encoded: " << encodedText << std::endl;// Base64 解码std::string decodedText = base64_decode(encodedText.c_str());std::cout << "Base64 Decoded: " << decodedText << std::endl;return 0;
}

编译

 g++ main.cpp -o main -lssl -lcrypto

运行结果如下:
在这里插入图片描述

四、源码地址

https://gitcode.net/arv002/qt/-/tree/master/Openssl/hash/base64

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

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

相关文章

C#中的扩展方法---Extension

C#中扩展方法是C# 3.0/.NET 3.x 新增特性&#xff0c;能够实现向现有类型中“添加”方法&#xff0c;以下主要介绍C#中扩展方法的声明及使用。 1、扩展方法的声明 扩展方法使能够向现有类型“添加”方法&#xff0c;而无需创建新的派生类型、重新编译或以其他方式修改原始类型…

xlua游戏热更新(C#访问lua)

xlua作为Unity资源热更新的重要解决方案api&#xff0c;在Tecent重多游戏中被采用&#xff0c;本文通过案例去讲解xlua代码结构层次。 /** Tencent is pleased to support the open source community by making xLua available.* Copyright (C) 2016 THL A29 Limited, a Tence…

快速走进通信世界 --- 基础知识扫盲

想不到吧&#xff0c;家人们&#xff0c;博主好久没来更新文章了&#xff0c;而且这次更新的是关于通信工程的文章。博主确实以前一直更新关于编程的文章&#xff0c;只不过最近在学习一些新的知识&#xff0c;以后有机会了我还是会继续更新一些编程技术文章的。不过每一门技术…

基于单片机设计的智能风扇(红外线无线控制开关调速定时)

一、项目介绍 在炎热的夏季&#xff0c;风扇成为人们室内生活中必不可少的电器产品。然而&#xff0c;传统的风扇控制方式存在一些不便之处&#xff0c;比如需要手动操作开关、无法远程控制和调速&#xff0c;以及缺乏定时功能等。为了解决这些问题&#xff0c;设计了一款基于…

如何用java写一个网站:从零搭建个性化网站

随着互联网的迅猛发展&#xff0c;Java作为一种强大而灵活的编程语言&#xff0c;为构建各类网站提供了丰富的解决方案。本文将探讨如何使用Java编写一个个性化网站&#xff0c;并通过具体实例进行深入分析。 第一步&#xff1a;选择适当的技术栈 在着手构建网站之前&#xff0…

【代码随想录】算法训练计划18

1、513. 找树左下角的值 题目&#xff1a; 给定一个二叉树的 根节点 root&#xff0c;请找出该二叉树的 最底层 最左边 节点的值。 假设二叉树中至少有一个节点。 思路&#xff1a; 递归&#xff0c;规则&#xff0c;基本可以自己写出来 var maxDepth int var res int fun…

深度学习之基于Django+Tensorflow商品识别管理系统

欢迎大家点赞、收藏、关注、评论啦 &#xff0c;由于篇幅有限&#xff0c;只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 项目简介 本系统是一个基于DjangoTensorflow的商品识别管理系统。通过深度学习技术&#xff0c;实现商品的自动识别…

Linux系统编程——文件的打开及创建

打开(open) 使用open函数需要包含以下三个头文件&#xff1a; #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> open的函数定义格式 int open(const char *pathname,int flags); int open(const char *pathname,int flags,mode_t mode…

CKA认证模块②-K8S企业运维和落地实战

CKA认证模块②-K8S企业运维和落地实战 Pod高级实战-Pod生命周期-启动钩子,停止钩子 Pod生命周期完整流程介绍 容器钩子; 容器探测; Pod重启策略; Pod的终止过程; Init容器; 初始化容器最佳实践 初始化容器与主容器区别是? init容器没有readinessProbe… [rootk8s-mast…

【机器学习】七、降维与度量学习

1. 维数灾难 样本的特征数称为维数&#xff08;dimensionality&#xff09;&#xff0c;当维数非常大时&#xff0c;也就是现在所说的维数灾难。 维数灾难具体表现在&#xff1a;在高维情形下&#xff0c;数据样本将变得十分稀疏&#xff0c;因为此时要满足训练样本为“密采样…

Postgres主键自增时重复键违反唯一约束

错误: 重复键违反唯一约束\"bue_new_copy1_pkey\"\n 详细&#xff1a;键值\"(id)(31)\"已经存在\n 新增的数据的id跟表里面的数据id重复了&#xff0c;这种一般是手动导入数据或者复制表等情况造成的&#xff0c;直接修改表的序列为当前最大的id&#xf…

25期代码随想录算法训练营第十四天 | 二叉树 | 递归遍历、迭代遍历

目录 递归遍历前序遍历中序遍历后序遍历 迭代遍历前序遍历中序遍历后序遍历 递归遍历 前序遍历 # Definition for a binary tree node. # class TreeNode: # def __init__(self, val0, leftNone, rightNone): # self.val val # self.left left # …