c++ DES ECB加密算法pkcs5padding填充实现

2024-02-11 12:18

本文主要是介绍c++ DES ECB加密算法pkcs5padding填充实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我主要的实现的pkcs5padding填充,其他填充方式自己实现就好了,都不难。

使用的是静态连接的方法,使用的是openssl的libcrypto库,网上一大堆,这里就不贴了。

windows库:https://download.csdn.net/download/suhiymof/11247287

头文件这样包含即可:

#ifdef WIN32
#include "des.h"
#pragma comment(lib, "libcrypt" )
#else
#include "openssl/des.h"
#endif //  WIN32

填充代码

//填充
char* PKCS5Padding(string strParams, uint32 unBlockSize)
{int nRaw_size = strParams.size();int i=0, j=nRaw_size/8+1, k=nRaw_size%8;int nPidding_size = 8 - k;char* szArray;szArray = (char *)malloc(nRaw_size + nPidding_size);memcpy(szArray, strParams.c_str(), nRaw_size);for (int i1=nRaw_size; i1<(nRaw_size+nPidding_size); i1++) {// PKCS5Padding 算法:szArray[i1] = nPidding_size;}return szArray;
}

windows加密:

//strParams:加密字符串
//szKey:加密的密码,我用的是八字节的密码
string Des_Encrypt(string strParams, char* szKey)
{using namespace CryptoPP;char szCommand[1024];memset(szCommand, 0, 1024);int nRaw_size = strParams.size();char* szArray = PKCS5Padding(strParams, DES::BLOCKSIZE);int j=nRaw_size/DES::BLOCKSIZE + 1;unsigned char key[DES::DEFAULT_KEYLENGTH];memcpy(key, szKey, DES::BLOCKSIZE);unsigned char szInput[ DES::BLOCKSIZE];unsigned char szOutput[ DES::BLOCKSIZE];for (int i = 0; i < j; ++i){memset(szInput, 0, 8);memset(szOutput, 0, 8);memcpy(szInput, szArray+(i*8), 8);DESEncryption encryption_DES;//因此,设置密匙。encryption_DES.SetKey(key, DES::KEYLENGTH );//进行加密encryption_DES.ProcessBlock( szInput, szOutput);memcpy(szCommand+(i*8), szOutput, 8);}int nLen = j * 8;//base64编码std::string strTmp = base64_encode(reinterpret_cast<const unsigned char*>(szCommand), nLen);return strTmp;
}

linux加密:

string Des_Encrypt(string strParams, char* szKey)
{char szCommand[1024];memset(szCommand, 0, 1024);int nRaw_size = strParams.size();char* szArray = PKCS5Padding(strParams, sizeof(DES_cblock));int j=nRaw_size/8 + 1;DES_cblock sKey;// = "12341234";memcpy(sKey, szKey, 8);const_DES_cblock sInput;DES_cblock sOutput;DES_key_schedule sSchedule;//转换成scheduleDES_set_key_unchecked(&sKey, &sSchedule); for (int i = 0; i < j; ++i){memset(sInput, 0, 8);memset(sOutput, 0, 8);memcpy(sInput, szArray+(i*8), 8);DES_ecb_encrypt(&sInput, &sOutput, &sSchedule, DES_ENCRYPT);memcpy(szCommand+(i*8), sOutput, 8);}int nLen = j * 8;//base64编码std::string strTmp = base64_encode(reinterpret_cast<const unsigned char*>(szCommand), nLen);return strTmp;
}

这里包含了base64位编码,这里贴一下源码

static const std::string base64_chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZ""abcdefghijklmnopqrstuvwxyz""0123456789+/";static inline bool is_base64(unsigned char c) {return (isalnum(c) || (c == '+') || (c == '/'));
}std::string base64_encode(unsigned char const* bytes_to_encode, unsigned int in_len) {printf("---%d\n", in_len);std::string ret;int i = 0;int j = 0;unsigned char char_array_3[3];unsigned char char_array_4[4];while (in_len--) {char_array_3[i++] = *(bytes_to_encode++);if (i == 3) {char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);char_array_4[3] = char_array_3[2] & 0x3f;for(i = 0; (i <4) ; i++)ret += base64_chars[char_array_4[i]];i = 0;}}if (i){for(j = i; j < 3; j++)char_array_3[j] = '\0';char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);char_array_4[3] = char_array_3[2] & 0x3f;for (j = 0; (j < i + 1); j++)ret += base64_chars[char_array_4[j]];while((i++ < 3))ret += '=';}//printf("---111%d, %s\n", ret.size(), ret.c_str());return ret;}std::string base64_decode(std::string const& encoded_string) {size_t in_len = encoded_string.size();int i = 0;int j = 0;int in_ = 0;unsigned char char_array_4[4], char_array_3[3];std::string ret;while (in_len-- && ( encoded_string[in_] != '=') && is_base64(encoded_string[in_])) {char_array_4[i++] = encoded_string[in_]; in_++;if (i ==4) {for (i = 0; i <4; i++)char_array_4[i] = base64_chars.find(char_array_4[i]);char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];for (i = 0; (i < 3); i++)ret += char_array_3[i];i = 0;}}if (i) {for (j = i; j <4; j++)char_array_4[j] = 0;for (j = 0; j <4; j++)char_array_4[j] = base64_chars.find(char_array_4[j]);char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];for (j = 0; (j < i - 1); j++) ret += char_array_3[j];}return ret;
}

解密:

windows

string Des_Decrypt(std::string strEncryptData, char* szKey)
{using namespace CryptoPP;//base64解码strEncryptData = base64_decode(strEncryptData);string strRet;std::vector<unsigned char> vecCleartext;int nRaw_size = strEncryptData.length();int j=nRaw_size/DES::BLOCKSIZE;unsigned char key[DES::DEFAULT_KEYLENGTH];memcpy(key, szKey, DES::BLOCKSIZE);unsigned char szInput[ DES::BLOCKSIZE];unsigned char szOutput[ DES::BLOCKSIZE];const char* szData = strEncryptData.c_str();//解密for (int i = 0; i < j; ++i){memset(szInput, 0, DES::BLOCKSIZE);memset(szOutput, 0, DES::BLOCKSIZE);memcpy(szInput, szData+(i*DES::BLOCKSIZE), DES::BLOCKSIZE);DESDecryption decryption_DES;//回忆一下之前的背景,对称加密算法需要一个密匙。加密和解密都会用到。//因此,设置密匙。decryption_DES.SetKey(key, DES::KEYLENGTH );//进行加密decryption_DES.ProcessBlock(szInput, szOutput);for (int k = 0; k < DES::BLOCKSIZE; k++)vecCleartext.push_back(szOutput[k]);}//填充数量int nPadNum = vecCleartext[nRaw_size-1];strRet.clear();strRet.assign(vecCleartext.begin(), vecCleartext.end()-nPadNum);return strRet;
}

linux:

string Des_Decrypt(std::string strEncryptData, char* szKey)
{//base64解码strEncryptData = base64_decode(strEncryptData);string strRet;std::vector<unsigned char> vecCleartext;DES_cblock sKey;memcpy(sKey, szKey, 8);DES_key_schedule sSchedule;//转换成scheduleDES_set_key_unchecked(&sKey, &sSchedule); const_DES_cblock sInput;DES_cblock sOutput;int nRaw_size = strEncryptData.length();int j=nRaw_size/8;const char* szData = strEncryptData.c_str();//解密unsigned char szTmp[8];for (int i = 0; i < j; ++i){memset(sInput, 0, 8);memset(sOutput, 0, 8);memcpy(sInput, szData+(i*8), 8);DES_ecb_encrypt(&sInput, &sOutput, &sSchedule, DES_DECRYPT);for (int k = 0; k < 8; k++)vecCleartext.push_back(sOutput[k]);}//填充数量int nPadNum = vecCleartext[nRaw_size-1];strRet.clear();strRet.assign(vecCleartext.begin(), vecCleartext.end()-nPadNum);return strRet;
}

 

这篇关于c++ DES ECB加密算法pkcs5padding填充实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Nginx更新SSL证书的实现步骤

《Nginx更新SSL证书的实现步骤》本文主要介绍了Nginx更新SSL证书的实现步骤,包括下载新证书、备份旧证书、配置新证书、验证配置及遇到问题时的解决方法,感兴趣的了解一下... 目录1 下载最新的SSL证书文件2 备份旧的SSL证书文件3 配置新证书4 验证配置5 遇到的http://www.cppc

Nginx之https证书配置实现

《Nginx之https证书配置实现》本文主要介绍了Nginx之https证书配置的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起... 目录背景介绍为什么不能部署在 IIS 或 NAT 设备上?具体实现证书获取nginx配置扩展结果验证

SpringBoot整合 Quartz实现定时推送实战指南

《SpringBoot整合Quartz实现定时推送实战指南》文章介绍了SpringBoot中使用Quartz动态定时任务和任务持久化实现多条不确定结束时间并提前N分钟推送的方案,本文结合实例代码给大... 目录前言一、Quartz 是什么?1、核心定位:解决什么问题?2、Quartz 核心组件二、使用步骤1

使用Redis实现会话管理的示例代码

《使用Redis实现会话管理的示例代码》文章介绍了如何使用Redis实现会话管理,包括会话的创建、读取、更新和删除操作,通过设置会话超时时间并重置,可以确保会话在用户持续活动期间不会过期,此外,展示了... 目录1. 会话管理的基本概念2. 使用Redis实现会话管理2.1 引入依赖2.2 会话管理基本操作

mybatis-plus分表实现案例(附示例代码)

《mybatis-plus分表实现案例(附示例代码)》MyBatis-Plus是一个MyBatis的增强工具,在MyBatis的基础上只做增强不做改变,为简化开发、提高效率而生,:本文主要介绍my... 目录文档说明数据库水平分表思路1. 为什么要水平分表2. 核心设计要点3.基于数据库水平分表注意事项示例

C#高效实现在Word文档中自动化创建图表的可视化方案

《C#高效实现在Word文档中自动化创建图表的可视化方案》本文将深入探讨如何利用C#,结合一款功能强大的第三方库,实现在Word文档中自动化创建图表,为你的数据呈现和报告生成提供一套实用且高效的解决方... 目录Word文档图表自动化:为什么选择C#?从零开始:C#实现Word文档图表的基本步骤深度优化:C

nginx跨域访问配置的几种方法实现

《nginx跨域访问配置的几种方法实现》本文详细介绍了Nginx跨域配置方法,包括基本配置、只允许指定域名、携带Cookie的跨域、动态设置允许的Origin、支持不同路径的跨域控制、静态资源跨域以及... 目录一、基本跨域配置二、只允许指定域名跨域三、完整示例四、配置后重载 nginx五、注意事项六、支持

Qt实现对Word网页的读取功能

《Qt实现对Word网页的读取功能》文章介绍了几种在Qt中实现Word文档(.docx/.doc)读写功能的方法,包括基于QAxObject的COM接口调用、DOCX模板替换及跨平台解决方案,重点讨论... 目录1. 核心实现方式2. 基于QAxObject的COM接口调用(Windows专用)2.1 环境

MySQL查看表的历史SQL的几种实现方法

《MySQL查看表的历史SQL的几种实现方法》:本文主要介绍多种查看MySQL表历史SQL的方法,包括通用查询日志、慢查询日志、performance_schema、binlog、第三方工具等,并... 目录mysql 查看某张表的历史SQL1.查看MySQL通用查询日志(需提前开启)2.查看慢查询日志3.

Java实现字符串大小写转换的常用方法

《Java实现字符串大小写转换的常用方法》在Java中,字符串大小写转换是文本处理的核心操作之一,Java提供了多种灵活的方式来实现大小写转换,适用于不同场景和需求,本文将全面解析大小写转换的各种方法... 目录前言核心转换方法1.String类的基础方法2. 考虑区域设置的转换3. 字符级别的转换高级转换