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

相关文章

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文

Golang如何对cron进行二次封装实现指定时间执行定时任务

《Golang如何对cron进行二次封装实现指定时间执行定时任务》:本文主要介绍Golang如何对cron进行二次封装实现指定时间执行定时任务问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录背景cron库下载代码示例【1】结构体定义【2】定时任务开启【3】使用示例【4】控制台输出总结背景

MySQL 获取字符串长度及注意事项

《MySQL获取字符串长度及注意事项》本文通过实例代码给大家介绍MySQL获取字符串长度及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 获取字符串长度详解 核心长度函数对比⚠️ 六大关键注意事项1. 字符编码决定字节长度2

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函