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

相关文章

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

python修改字符串值的三种方法

《python修改字符串值的三种方法》本文主要介绍了python修改字符串值的三种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录第一种方法:第二种方法:第三种方法:在python中,字符串对象是不可变类型,所以我们没办法直接

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Java中的密码加密方式

《Java中的密码加密方式》文章介绍了Java中使用MD5算法对密码进行加密的方法,以及如何通过加盐和多重加密来提高密码的安全性,MD5是一种不可逆的哈希算法,适合用于存储密码,因为其输出的摘要长度固... 目录Java的密码加密方式密码加密一般的应用方式是总结Java的密码加密方式密码加密【这里采用的

C#中字符串分割的多种方式

《C#中字符串分割的多种方式》在C#编程语言中,字符串处理是日常开发中不可或缺的一部分,字符串分割是处理文本数据时常用的操作,它允许我们将一个长字符串分解成多个子字符串,本文给大家介绍了C#中字符串分... 目录1. 使用 string.Split2. 使用正则表达式 (Regex.Split)3. 使用

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

使用Python制作一个PDF批量加密工具

《使用Python制作一个PDF批量加密工具》PDF批量加密‌是一种保护PDF文件安全性的方法,通过为多个PDF文件设置相同的密码,防止未经授权的用户访问这些文件,下面我们来看看如何使用Python制... 目录1.简介2.运行效果3.相关源码1.简介一个python写的PDF批量加密工具。PDF批量加密

Java中JSON字符串反序列化(动态泛型)

《Java中JSON字符串反序列化(动态泛型)》文章讨论了在定时任务中使用反射调用目标对象时处理动态参数的问题,通过将方法参数存储为JSON字符串并进行反序列化,可以实现动态调用,然而,这种方式容易导... 需求:定时任务扫描,反射调用目标对象,但是,方法的传参不是固定的。方案一:将方法参数存成jsON字