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),直接操纵和返回字符串的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:https://blog.csdn.net/chuachua66/article/details/50583352
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/169645

相关文章

golang float和科学计数法转字符串的实现方式

《golangfloat和科学计数法转字符串的实现方式》:本文主要介绍golangfloat和科学计数法转字符串的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望... 目录golang float和科学计数法转字符串需要对float转字符串做处理总结golang float

Python如何判断字符串中是否包含特殊字符并替换

《Python如何判断字符串中是否包含特殊字符并替换》这篇文章主要为大家详细介绍了如何使用Python实现判断字符串中是否包含特殊字符并使用空字符串替换掉,文中的示例代码讲解详细,感兴趣的小伙伴可以了... 目录python判断字符串中是否包含特殊字符方法一:使用正则表达式方法二:手动检查特定字符Pytho

C++ HTTP框架推荐(特点及优势)

《C++HTTP框架推荐(特点及优势)》:本文主要介绍C++HTTP框架推荐的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Crow2. Drogon3. Pistache4. cpp-httplib5. Beast (Boos

使用Java实现Navicat密码的加密与解密的代码解析

《使用Java实现Navicat密码的加密与解密的代码解析》:本文主要介绍使用Java实现Navicat密码的加密与解密,通过本文,我们了解了如何利用Java语言实现对Navicat保存的数据库密... 目录一、背景介绍二、环境准备三、代码解析四、核心代码展示五、总结在日常开发过程中,我们有时需要处理各种软

MySQL 字符串截取函数及用法详解

《MySQL字符串截取函数及用法详解》在MySQL中,字符串截取是常见的操作,主要用于从字符串中提取特定部分,MySQL提供了多种函数来实现这一功能,包括LEFT()、RIGHT()、SUBST... 目录mysql 字符串截取函数详解RIGHT(str, length):从右侧截取指定长度的字符SUBST

Python将字符串转换为小写字母的几种常用方法

《Python将字符串转换为小写字母的几种常用方法》:本文主要介绍Python中将字符串大写字母转小写的四种方法:lower()方法简洁高效,手动ASCII转换灵活可控,str.translate... 目录一、使用内置方法 lower()(最简单)二、手动遍历 + ASCII 码转换三、使用 str.tr

Java如何用乘号来重复字符串的功能

《Java如何用乘号来重复字符串的功能》:本文主要介绍Java使用乘号来重复字符串的功能,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java乘号来重复字符串的功能1、利用循环2、使用StringBuilder3、采用 Java 11 引入的String.rep

C++类和对象之初始化列表的使用方式

《C++类和对象之初始化列表的使用方式》:本文主要介绍C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C++初始化列表详解:性能优化与正确实践什么是初始化列表?初始化列表的三大核心作用1. 性能优化:避免不必要的赋值操作2. 强

C++迭代器失效的避坑指南

《C++迭代器失效的避坑指南》在C++中,迭代器(iterator)是一种类似指针的对象,用于遍历STL容器(如vector、list、map等),迭代器失效是指在对容器进行某些操作后... 目录1. 什么是迭代器失效?2. 哪些操作会导致迭代器失效?2.1 vector 的插入操作(push_back,

一文详解如何在Vue3中封装API请求

《一文详解如何在Vue3中封装API请求》在现代前端开发中,API请求是不可避免的一部分,尤其是与后端交互时,下面我们来看看如何在Vue3项目中封装API请求,让你在实现功能时更加高效吧... 目录为什么要封装API请求1. vue 3项目结构2. 安装axIOS3. 创建API封装模块4. 封装API请求