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

相关文章

Python使用自带的base64库进行base64编码和解码

《Python使用自带的base64库进行base64编码和解码》在Python中,处理数据的编码和解码是数据传输和存储中非常普遍的需求,其中,Base64是一种常用的编码方案,本文我将详细介绍如何使... 目录引言使用python的base64库进行编码和解码编码函数解码函数Base64编码的应用场景注意

C#数据结构之字符串(string)详解

《C#数据结构之字符串(string)详解》:本文主要介绍C#数据结构之字符串(string),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录转义字符序列字符串的创建字符串的声明null字符串与空字符串重复单字符字符串的构造字符串的属性和常用方法属性常用方法总结摘

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

SpringBoot中封装Cors自动配置方式

《SpringBoot中封装Cors自动配置方式》:本文主要介绍SpringBoot中封装Cors自动配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot封装Cors自动配置背景实现步骤1. 创建 GlobalCorsProperties

C++ 中的 if-constexpr语法和作用

《C++中的if-constexpr语法和作用》if-constexpr语法是C++17引入的新语法特性,也被称为常量if表达式或静态if(staticif),:本文主要介绍C++中的if-c... 目录1 if-constexpr 语法1.1 基本语法1.2 扩展说明1.2.1 条件表达式1.2.2 fa

python中字符串拼接的几种方法及优缺点对比详解

《python中字符串拼接的几种方法及优缺点对比详解》在Python中,字符串拼接是常见的操作,Python提供了多种方法来拼接字符串,每种方法有其优缺点和适用场景,以下是几种常见的字符串拼接方法,需... 目录1. 使用 + 运算符示例:优缺点:2. 使用&nbsjsp;join() 方法示例:优缺点:3

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

java字符串数字补齐位数详解

《java字符串数字补齐位数详解》:本文主要介绍java字符串数字补齐位数,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java字符串数字补齐位数一、使用String.format()方法二、Apache Commons Lang库方法三、Java 11+的St

C++从序列容器中删除元素的四种方法

《C++从序列容器中删除元素的四种方法》删除元素的方法在序列容器和关联容器之间是非常不同的,在序列容器中,vector和string是最常用的,但这里也会介绍deque和list以供全面了解,尽管在一... 目录一、简介二、移除给定位置的元素三、移除与某个值相等的元素3.1、序列容器vector、deque