C++封装AES加密类,(ECB + BASE64 + pkcs5padding),直接操纵和返回字符串

本文主要是介绍C++封装AES加密类,(ECB + BASE64 + pkcs5padding),直接操纵和返回字符串,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前几天需要使用AES加密,其他语言都是现成的类库,new 个实例 + 传个参数 = 搞定!

C++的也有现成的crypto++之类的库,不过太大了,一个库就40多M,没法接受,只能自己寻找资料自力更生(我们就是原始人。。。。)

其实不复杂,算法部分到处都有,但是麻烦就在最后一公里上,怎么弄个字符串传进去再传出来一个字符串,困扰了一天!

首先需要搞清楚几个概念:

AES加密分几种方法:ECB和其他几种;

    ECB的方式不需要IV(初始向量:就是再秘钥基础上再偏移一下更难破解),只需要有秘钥就可以了;

    其他方式都需要有秘钥和IV。

其次,AES加密需要整组整组的加密,也就是说少于16个字符需要补足16个字符,正好等于16个字符需要再额外加上16个字符,这个东西叫padding,有None, pkcs5padding, pkcs7padding等等种类,具体可以看看这哥们的文章,C#和JAVA支持的方式不一样,这也会导致不同语言加的密解不开。。。。

 

最后,加密算法操作的和返回的都是unsigned char数组,而我们需要通常是string 进,string出。而且出来的都是”XDFSLDJFF==“ 这样的东西,这个东西就是BASE64编码后的东西了。

 

我就是把这些麻烦的东西汇总了一下,写个能操纵字符串的类,方便以后的同学们使用!

(我是在这个在线工具对比的结果)

 

具体的工程在我的资源里可以下载微笑

 

感谢@yakovchang2017 @ruyi366 的反馈,我当时只做了16个字符内的,超过16字符确实是会报错,下面的改了下,把StringToHex加一个destlen就好了。

没想到还能帮到一些朋友,如果有需要的朋友直接用下面的代码覆盖aeshelper.cpp就可以了。带来误解也给大家说声抱歉!:)

 

 

#include "AesHelper.h"
#include "aes.h"
#include "zbase64.h"CAesHelper::CAesHelper(void)
{
}CAesHelper::~CAesHelper(void)
{
}void CAesHelper::StringToHex(const char* pSrc, unsigned char* pDest, int nDestLen)
{int nSrcLen = 0;if( pSrc != 0 ){nSrcLen = strlen(pSrc);memcpy(pDest, pSrc, nSrcLen > nDestLen ? nDestLen : nSrcLen);}Padding(pDest, nSrcLen > nDestLen ? nDestLen : nSrcLen);
}void CAesHelper::Padding( unsigned char* pSrc, int nSrcLen )
{if( nSrcLen < KEYCODELENGTH ){unsigned char ucPad = KEYCODELENGTH - nSrcLen;for( int nID = KEYCODELENGTH; nID > nSrcLen; --nID ){pSrc[nID - 1] = ucPad;}}
}std::string CAesHelper::Encrypt( std::string strSrc, std::string strKey )
{ZBase64 tool;aes_context ctx;const char* pSrc = 0;const char* pTmpSrc = 0;unsigned char* pDest = 0;unsigned char* pTmpDest = 0;int nSrcLen = 0;int nDestLen = 0;unsigned char buf[KEYCODELENGTH];unsigned char key[KEYCODELENGTH];StringToHex( strKey.c_str(), key, KEYCODELENGTH );aes_set_key( &ctx, key, 128);pSrc = strSrc.c_str();nSrcLen = strSrc.size();nDestLen = (nSrcLen / KEYCODELENGTH) * KEYCODELENGTH + KEYCODELENGTH;pDest = new unsigned char[nDestLen];memset( pDest, 0, nDestLen );pTmpSrc = pSrc;pTmpDest = pDest;while( (pTmpSrc - pSrc) < nSrcLen ){StringToHex(pTmpSrc, buf, KEYCODELENGTH);aes_encrypt( &ctx, buf, buf );memcpy( pTmpDest, buf, KEYCODELENGTH );pTmpSrc += KEYCODELENGTH;pTmpDest += KEYCODELENGTH;}if( (pTmpDest - pDest) < nDestLen )	// if the source size % KEYCODELENGTH == 0 then need to add an extra buffer {StringToHex(0, buf, KEYCODELENGTH);aes_encrypt( &ctx, buf, buf );memcpy( pTmpDest, buf, KEYCODELENGTH );}std::string strRet = tool.Encode(pDest, nDestLen);delete [] pDest;return strRet;
}std::string CAesHelper::Decrypt( std::string strSrc, std::string strKey )
{ZBase64 tool;aes_context ctx;const char* pSrc = 0;const char* pTmpSrc = 0;unsigned char* pDest = 0;unsigned char* pTmpDest = 0;int nSrcLen = 0;int nDestLen = 0;unsigned char buf[KEYCODELENGTH];unsigned char key[KEYCODELENGTH];StringToHex(strKey.c_str(), key, KEYCODELENGTH);aes_set_key( &ctx, key, 128);std::string strSrcHex = tool.Decode(strSrc.c_str(), strSrc.size(), nSrcLen);pSrc = strSrcHex.data();nDestLen = nSrcLen;pDest = new unsigned char[nDestLen];memset( pDest, 0, nDestLen );pTmpSrc = pSrc;pTmpDest = pDest;while( (pTmpSrc - pSrc) < nSrcLen ){memcpy(buf, pTmpSrc, KEYCODELENGTH);aes_decrypt( &ctx, buf, buf );memcpy(pTmpDest, buf, KEYCODELENGTH);pTmpSrc += KEYCODELENGTH;pTmpDest += KEYCODELENGTH;}unsigned char ucTest = 0;while(ucTest = *(pTmpDest - 1)){if( ucTest > 0 && ucTest <= 0x10 )*(pTmpDest-1) = 0;elsebreak;pTmpDest--;}std::string strRet = (char*)pDest;delete [] pDest;return strRet;
}

 

 

 

 

 

这篇关于C++封装AES加密类,(ECB + BASE64 + pkcs5padding),直接操纵和返回字符串的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/169645

相关文章

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

06 C++Lambda表达式

lambda表达式的定义 没有显式模版形参的lambda表达式 [捕获] 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 有显式模版形参的lambda表达式 [捕获] <模版形参> 模版约束 前属性 (形参列表) 说明符 异常 后属性 尾随类型 约束 {函数体} 含义 捕获:包含零个或者多个捕获符的逗号分隔列表 模板形参:用于泛型lambda提供个模板形参的名

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

【C++高阶】C++类型转换全攻略:深入理解并高效应用

📝个人主页🌹:Eternity._ ⏩收录专栏⏪:C++ “ 登神长阶 ” 🤡往期回顾🤡:C++ 智能指针 🌹🌹期待您的关注 🌹🌹 ❀C++的类型转换 📒1. C语言中的类型转换📚2. C++强制类型转换⛰️static_cast🌞reinterpret_cast⭐const_cast🍁dynamic_cast 📜3. C++强制类型转换的原因📝

BUUCTF(34)特殊的 BASE64

使用pycharm时,如果想把代码撤销到之前的状态可以用 Ctrl+z 如果不小心撤销多了,可以用 Ctrl+Shift+Z 还原, 别傻傻的重新敲了 BUUCTF在线评测 (buuoj.cn) 查看字符串,想到base64的变表 这里用的c++的标准程序库中的string,头文件是#include<string> 这是base64的加密函数 std::string

C++——stack、queue的实现及deque的介绍

目录 1.stack与queue的实现 1.1stack的实现  1.2 queue的实现 2.重温vector、list、stack、queue的介绍 2.1 STL标准库中stack和queue的底层结构  3.deque的简单介绍 3.1为什么选择deque作为stack和queue的底层默认容器  3.2 STL中对stack与queue的模拟实现 ①stack模拟实现

c++的初始化列表与const成员

初始化列表与const成员 const成员 使用const修饰的类、结构、联合的成员变量,在类对象创建完成前一定要初始化。 不能在构造函数中初始化const成员,因为执行构造函数时,类对象已经创建完成,只有类对象创建完成才能调用成员函数,构造函数虽然特殊但也是成员函数。 在定义const成员时进行初始化,该语法只有在C11语法标准下才支持。 初始化列表 在构造函数小括号后面,主要用于给