不使用任何库,用C++写出AES加密算法

时间: 2024-06-09 11:06:11 浏览: 7
AES(Advanced Encryption Standard)是一种对称加密算法,它使用相同的密钥用于加密和解密数据。下面是一个简单的C++实现AES加密算法的示例代码。 ```c++ #include <iostream> #include <cstring> // AES加密函数 void aes_encrypt(unsigned char* message, unsigned char* key, int message_size, int key_size) { // 通过密钥计算出轮密钥 unsigned char round_key[176]; key_expansion(key, round_key, key_size); // 将明文按照每个块128位进行分组,每个块内部进行AES加密 for (int i = 0; i < message_size; i += 16) { unsigned char block[16]; memcpy(block, message + i, 16); aes_encrypt_block(block, round_key); memcpy(message + i, block, 16); } } // AES加密块函数 void aes_encrypt_block(unsigned char* block, unsigned char* round_key) { // 轮密钥加 add_round_key(block, round_key, 0); // 9轮迭代 for (int round = 1; round <= 9; round++) { sub_bytes(block); shift_rows(block); mix_columns(block); add_round_key(block, round_key, round * 16); } // 最后一轮 sub_bytes(block); shift_rows(block); add_round_key(block, round_key, 160); } // 密钥扩展函数 void key_expansion(unsigned char* key, unsigned char* round_key, int key_size) { // 初始轮密钥即为密钥 memcpy(round_key, key, key_size); // 轮密钥扩展 for (int i = key_size; i < 176; i += 4) { unsigned char temp[4]; memcpy(temp, round_key + i - 4, 4); if (i % key_size == 0) { // 循环左移一位 unsigned char t = temp[0]; temp[0] = temp[1]; temp[1] = temp[2]; temp[2] = temp[3]; temp[3] = t; // 字节代换 temp[0] = s_box[temp[0]]; temp[1] = s_box[temp[1]]; temp[2] = s_box[temp[2]]; temp[3] = s_box[temp[3]]; // 异或上轮常数 temp[0] ^= r_con[i / key_size]; } else if (key_size > 24 && i % key_size == 16) { // 字节代换 temp[0] = s_box[temp[0]]; temp[1] = s_box[temp[1]]; temp[2] = s_box[temp[2]]; temp[3] = s_box[temp[3]]; } // 异或上前4个字节 temp[0] ^= round_key[i - key_size]; temp[1] ^= round_key[i - key_size + 1]; temp[2] ^= round_key[i - key_size + 2]; temp[3] ^= round_key[i - key_size + 3]; // 将扩展后的密钥放入轮密钥中 memcpy(round_key + i, temp, 4); } } // 轮密钥加函数 void add_round_key(unsigned char* block, unsigned char* round_key, int offset) { for (int i = 0; i < 16; i++) { block[i] ^= round_key[i + offset]; } } // 字节代换函数 void sub_bytes(unsigned char* block) { for (int i = 0; i < 16; i++) { block[i] = s_box[block[i]]; } } // 行移位函数 void shift_rows(unsigned char* block) { unsigned char temp[16]; memcpy(temp, block, 16); // 第1行不变 block[1] = temp[5]; block[2] = temp[10]; block[3] = temp[15]; // 第2行循环左移1位 block[5] = temp[9]; block[6] = temp[14]; block[7] = temp[3]; block[9] = temp[13]; block[10] = temp[2]; block[11] = temp[7]; // 第3行循环左移2位 block[15] = temp[6]; block[14] = temp[1]; block[13] = temp[12]; block[12] = temp[8]; block[11] = temp[4]; block[10] = temp[0]; } // 列混淆函数 void mix_columns(unsigned char* block) { for (int i = 0; i < 4; i++) { unsigned char a[4], b[4]; memcpy(a, block + i * 4, 4); memcpy(b, block + i * 4, 4); // 列混淆矩阵 block[i * 4] = gf_mul(0x02, a[0]) ^ gf_mul(0x03, a[1]) ^ gf_mul(0x01, a[2]) ^ gf_mul(0x01, a[3]); block[i * 4 + 1] = gf_mul(0x01, b[0]) ^ gf_mul(0x02, b[1]) ^ gf_mul(0x03, b[2]) ^ gf_mul(0x01, b[3]); block[i * 4 + 2] = gf_mul(0x01, a[0]) ^ gf_mul(0x01, a[1]) ^ gf_mul(0x02, a[2]) ^ gf_mul(0x03, a[3]); block[i * 4 + 3] = gf_mul(0x03, b[0]) ^ gf_mul(0x01, b[1]) ^ gf_mul(0x01, b[2]) ^ gf_mul(0x02, b[3]); } } // 有限域乘法函数 unsigned char gf_mul(unsigned char a, unsigned char b) { unsigned char p = 0; for (int i = 0; i < 8; i++) { if (b & 1) { p ^= a; } bool hi_bit_set = a & 0x80; a <<= 1; if (hi_bit_set) { a ^= 0x1b; } b >>= 1; } return p; } // S盒 const unsigned char s_box[256] = { 0x63,0x7c,0x77,0x7b,0xf2,0x6b,0x6f,0xc5,0x30,0x01,0x67,0x2b,0xfe,0xd7,0xab,0x76, 0xca,0x82,0xc9,0x7d,0xfa,0x59,0x47,0xf0,0xad,0xd4,0xa2,0xaf,0x9c,0xa4,0x72,0xc0, 0xb7,0xfd,0x93,0x26,0x36,0x3f,0xf7,0xcc,0x34,0xa5,0xe5,0xf1,0x71,0xd8,0x31,0x15, 0x04,0xc7,0x23,0xc3,0x18,0x96,0x05,0x9a,0x07,0x12,0x80,0xe2,0xeb,0x27,0xb2,0x75, 0x09,0x83,0x2c,0x1a,0x1b,0x6e,0x5a,0xa0,0x52,0x3b,0xd6,0xb3,0x29,0xe3,0x2f,0x84, 0x53,0xd1,0x00,0xed,0x20,0xfc,0xb1,0x5b,0x6a,0xcb,0xbe,0x39,0x4a,0x4c,0x58,0xcf, 0xd0,0xef,0xaa,0xfb,0x43,0x4d,0x33,0x85,0x45,0xf9,0x02,0x7f,0x50,0x3c,0x9f,0xa8, 0x51,0xa3,0x40,0x8f,0x92,0x9d,0x38,0xf5,0xbc,0xb6,0xda,0x21,0x10,0xff,0xf3,0xd2, 0xcd,0x0c,0x13,0xec,0x5f,0x97,0x44,0x17,0xc4,0xa7,0x7e,0x3d,0x64,0x5d,0x19,0x73, 0x60,0x81,0x4f,0xdc,0x22,0x2a,0x90,0x88,0x46,0xee,0xb8,0x14,0xde,0x5e,0x0b,0xdb, 0xe0,0x32,0x3a,0x0a,0x49,0x06,0x24,0x5c,0xc2,0xd3,0xac,0x62,0x91,0x95,0xe4,0x79, 0xe7,0xc8,0x37,0x6d,0x8d,0xd5,0x4e,0xa9,0x6c,0x56,0xf4,0xea,0x65,0x7a,0xae,0x08, 0xba,0x78,0x25,0x2e,0x1c,0xa6,0xb4,0xc6,0xe8,0xdd,0x74,0x1f,0x4b,0xbd,0x8b,0x8a, 0x70,0x3e,0xb5,0x66,0x48,0x03,0xf6,0x0e,0x61,0x35,0x57,0xb9,0x86,0xc1,0x1d,0x9e, 0xe1,0xf8,0x98,0x11,0x69,0xd9,0x8e,0x94,0x9b,0x1e,0x87,0xe9,0xce,0x55,0x28,0xdf, 0x8c,0xa1,0x89,0x0d,0xbf,0xe6,0x42,0x68,0x41,0x99,0x2d,0x0f,0xb0,0x54,0xbb,0x16 }; // 轮常数 const unsigned char r_con[10] = { 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36 }; int main() { // 明文和密钥 unsigned char message[] = "hello world"; unsigned char key[] = "secret key"; // 加密 aes_encrypt(message, key, strlen((char*)message), strlen((char*)key)); // 输出密文 std::cout << "Ciphertext: "; for (int i = 0; i < strlen((char*)message); i++) { printf("%02x ", message[i]); } std::cout << std::endl; return 0; } ```

相关推荐

最新推荐

recommend-type

android使用AES加密和解密文件实例代码

AES 加密算法使用同一个密钥进行加密和解密,可以确保数据的安全性。 Android 中的 AES 加密 在 Android 应用程序中,使用 AES 加密需要使用 Java 加密架构(Java Cryptography Architecture,JCA)提供的加密 API...
recommend-type

c++代码实现tea加密算法的实例详解

本文将详细介绍如何使用C++实现TEA加密算法,并探讨其在实际应用中的注意事项。 TEA加密算法的核心在于其加密过程,由`tea_encrypt`和`tea_decrypt`两个函数实现。这两个函数分别用于加密和解密,它们接受两个32位...
recommend-type

C++不使用变量求字符串长度strlen函数的实现方法

在标准库`&lt;cstring&gt;`中定义,`strlen`函数通常的使用方式是`strlen("example string")`,这将返回`"example string"`中的字符数,不包括结束的空字符。然而,有时候在面试或者编程挑战中,我们可能被要求不使用额外...
recommend-type

使用C++实现全排列算法的方法详解

总的来说,使用C++实现全排列算法涉及对递增进位制和递减进位制数的理解与操作,通过映射和还原过程生成所有可能的排列。这种算法不仅在编程竞赛和算法设计中常见,也是解决实际问题如密码学、组合优化等领域的重要...
recommend-type

使用c++编写和使用.so动态链接库

1,使用 c 生成动态链接库mylib.so的简单示例  声明文件mylib.h   #ifndef __MY_LIB_H__ #define __MY_LIB_H__ void foobar(int i); #endif /* __MY_LIB_H__ */  实现文件...
recommend-type

VMP技术解析:Handle块优化与壳模板初始化

"这篇学习笔记主要探讨了VMP(Virtual Machine Protect,虚拟机保护)技术在Handle块优化和壳模板初始化方面的应用。作者参考了看雪论坛上的多个资源,包括关于VMP还原、汇编指令的OpCode快速入门以及X86指令编码内幕的相关文章,深入理解VMP的工作原理和技巧。" 在VMP技术中,Handle块是虚拟机执行的关键部分,它包含了用于执行被保护程序的指令序列。在本篇笔记中,作者详细介绍了Handle块的优化过程,包括如何删除不使用的代码段以及如何通过指令变形和等价替换来提高壳模板的安全性。例如,常见的指令优化可能将`jmp`指令替换为`push+retn`或者`lea+jmp`,或者将`lodsbyteptrds:[esi]`优化为`moval,[esi]+addesi,1`等,这些变换旨在混淆原始代码,增加反逆向工程的难度。 在壳模板初始化阶段,作者提到了1.10和1.21两个版本的区别,其中1.21版本增加了`Encodingofap-code`保护,增强了加密效果。在未加密时,代码可能呈现出特定的模式,而加密后,这些模式会被混淆,使分析更加困难。 笔记中还提到,VMP会使用一个名为`ESIResults`的数组来标记Handle块中的指令是否被使用,值为0表示未使用,1表示使用。这为删除不必要的代码提供了依据。此外,通过循环遍历特定的Handle块,并依据某种规律(如`v227&0xFFFFFF00==0xFACE0000`)进行匹配,可以找到需要处理的指令,如`push0xFACE0002`和`movedi,0xFACE0003`,然后将其替换为安全的重定位值或虚拟机上下文。 在结构体使用方面,笔记指出壳模板和用户代码都会通过`Vmp_AllDisassembly`函数进行解析,而且0x8和0x10字段通常都指向相同的结构体。作者还提到了根据`pNtHeader_OptionalHeader.Magic`筛选`ESI_Matching_Array`数组的步骤,这可能是为了进一步确定虚拟机上下文的设置。 这篇笔记深入解析了VMP技术在代码保护中的应用,涉及汇编指令的优化、Handle块的处理以及壳模板的初始化,对于理解反逆向工程技术以及软件保护策略有着重要的参考价值。
recommend-type

管理建模和仿真的文件

管理Boualem Benatallah引用此版本:布阿利姆·贝纳塔拉。管理建模和仿真。约瑟夫-傅立叶大学-格勒诺布尔第一大学,1996年。法语。NNT:电话:00345357HAL ID:电话:00345357https://theses.hal.science/tel-003453572008年12月9日提交HAL是一个多学科的开放存取档案馆,用于存放和传播科学研究论文,无论它们是否被公开。论文可以来自法国或国外的教学和研究机构,也可以来自公共或私人研究中心。L’archive ouverte pluridisciplinaire
recommend-type

【进阶】音频处理基础:使用Librosa

![【进阶】音频处理基础:使用Librosa](https://picx.zhimg.com/80/v2-a39e5c9bff1d920097341591ca8a2dfe_1440w.webp?source=1def8aca) # 2.1 Librosa库的安装和导入 Librosa库是一个用于音频处理的Python库。要安装Librosa库,请在命令行中输入以下命令: ``` pip install librosa ``` 安装完成后,可以通过以下方式导入Librosa库: ```python import librosa ``` 导入Librosa库后,就可以使用其提供的各种函数
recommend-type

python中字典转换成json

在Python中,你可以使用`json`模块将字典转换为JSON格式的字符串。下面是一个简单的示例: ```python import json # 假设我们有一个字典 dict_data = { "name": "John", "age": 30, "city": "New York" } # 使用json.dumps()函数将字典转换为JSON json_string = json.dumps(dict_data) print(json_string) # 输出:{"name": "John", "age": 30, "city": "New York"}
recommend-type

C++ Primer 第四版更新:现代编程风格与标准库

"Cpp Primer第四版中文版(电子版)1" 本书《Cpp Primer》第四版是一本深入浅出介绍C++编程语言的教程,旨在帮助初学者和有经验的程序员掌握现代C++编程技巧。作者在这一版中进行了重大更新,以适应C++语言的发展趋势,特别是强调使用标准库来提高编程效率。书中不再过于关注底层编程技术,而是将重点放在了标准库的运用上。 第四版的主要改动包括: 1. 内容重组:为了反映现代C++编程的最佳实践,书中对语言主题的顺序进行了调整,使得学习路径更加顺畅。 2. 添加辅助学习工具:每章增设了“小结”和“术语”部分,帮助读者回顾和巩固关键概念。此外,重要术语以黑体突出,已熟悉的术语以楷体呈现,以便读者识别。 3. 特殊标注:用特定版式标注关键信息,提醒读者注意语言特性,避免常见错误,强调良好编程习惯,同时提供通用的使用技巧。 4. 前后交叉引用:增加引用以帮助读者理解概念之间的联系。 5. 额外讨论和解释:针对复杂概念和初学者常遇到的问题,进行深入解析。 6. 大量示例:提供丰富的代码示例,所有源代码都可以在线获取,便于读者实践和学习。 本书保留了前几版的核心特色,即以实例教学,通过解释和展示语言特性来帮助读者掌握C++。作者的目标是创作一本清晰、全面、准确的教程,让读者在编写程序的过程中学习C++,同时也展示了如何有效地利用这门语言。 《Cpp Primer》第四版不仅适合C++初学者,也适合想要更新C++知识的老手,它全面覆盖了C++语言的各个方面,包括基础语法、类、模板、STL(Standard Template Library)等,同时引入了现代C++的特性,如智能指针、RAII(Resource Acquisition Is Initialization)、lambda表达式等,使读者能够跟上C++语言的发展步伐,提升编程技能。