c++中如何实现简单的文件加密器_c++异或逻辑加密文件教程【详解】

12次阅读

异或加密最简单因满足a^b^b==a,可流式逐块加解密;但非密码学安全,仅适用于低敏感场景,需避免内存溢出、符号扩展、硬编码密钥等问题。

c++中如何实现简单的文件加密器_c++异或逻辑加密文件教程【详解】

为什么用异或(XOR)做文件加密最简单

因为异或满足 a ^ b ^ b == a,同一密钥反复异或即可加解密——无需额外算法库、不依赖第三方、不用处理 padding 或块模式。但注意:这只是“混淆”,不是密码学安全的加密,仅适用于临时保护明文配置、日志片段等低敏感场景。

常见错误是直接对整个文件一次性读入内存再异或,导致大文件(如 >500MB)崩溃或卡死;另一个坑是密钥长度为 1 时误用 char 类型导致符号扩展,比如密钥值 0xFF 被解释成 -1,异或结果错乱。

  • 密钥建议用 unsigned charstd::uint8_t 存储,避免符号问题
  • 必须逐块读写(如 4KB~64KB),别用 std::ifstream::rdbuf() 一次性加载
  • 不要把密钥硬编码在可执行文件里——哪怕只是 demo,也该从命令行或环境变量传入

std::ifstreamstd::ofstream 实现流式异或

核心是用 read()/write() 配合 gcount() 处理每次实际读取字节数,避免因文件末尾不足缓冲区大小导致的覆盖或丢字节

int xor_encrypt_file(const std::String& input_path, const std::string& output_path, uint8_t key) {     std::ifstream fin(input_path, std::ios::binary);     std::ofstream fout(output_path, std::ios::binary);     if (!fin || !fout) return -1;      constexpr size_t BUFFER_SIZE = 8192;     std::vector buffer(BUFFER_SIZE);      while (fin.read(reinterpret_cast(buffer.data()), BUFFER_SIZE)) {         size_t bytes_read = fin.gcount();         for (size_t i = 0; i < bytes_read; ++i) {             buffer[i] ^= key;         }         fout.write(reinterpret_cast(buffer.data()), bytes_read);     }      // 处理剩余不足 BUFFER_SIZE 的部分     if (fin.gcount() > 0) {         size_t bytes_read = fin.gcount();         for (size_t i = 0; i < bytes_read; ++i) {             buffer[i] ^= key;         }         fout.write(reinterpret_cast(buffer.data()), bytes_read);     }      return 0; }

这段代码能安全处理任意大小文件。关键点:gcount() 必须在每次 read() 后立即读取,不能用 fin.tellg() 推算——后者在二进制流中不可靠。

立即学习C++免费学习笔记(深入)”;

支持多字节密钥的循环异或(更抗统计分析)

单字节密钥易被频率分析破解(比如大量 0x00 字节异或后变成固定值)。换成字符串密钥(如 "mykey123"),按位置循环使用每个字节,提升混淆强度。

注意点:

  • 密钥字符串需转为 std::vector,避免 std::string 中的 提前截断
  • 索引用 i % key.size(),但要确保 key 非空,否则除零崩溃
  • 不要用 std::string::c_str() 直接取指针——遇到 就停了
int xor_encrypt_file_multikey(const std::string& input_path, const std::string& output_path, const std::string& key_str) {     if (key_str.empty()) return -1;     std::vector key(key_str.begin(), key_str.end());      std::ifstream fin(input_path, std::ios::binary);     std::ofstream fout(output_path, std::ios::binary);     if (!fin || !fout) return -1;      constexpr size_t BUFFER_SIZE = 8192;     std::vector buffer(BUFFER_SIZE);      size_t key_idx = 0;     while (fin.read(reinterpret_cast(buffer.data()), BUFFER_SIZE)) {         size_t bytes_read = fin.gcount();         for (size_t i = 0; i < bytes_read; ++i) {             buffer[i] ^= key[key_idx++ % key.size()];         }         fout.write(reinterpret_cast(buffer.data()), bytes_read);     }      if (fin.gcount() > 0) {         size_t bytes_read = fin.gcount();         for (size_t i = 0; i < bytes_read; ++i) {             buffer[i] ^= key[key_idx++ % key.size()];         }         fout.write(reinterpret_cast(buffer.data()), bytes_read);     }      return 0; }

命令行调用与常见报错处理

编译时加 -std=c++17(需要 std::bytestd::uint8_t),运行时若提示 No such file or Directory,先确认路径是否含空格或中文——C++ 标准库对非 ASCII 路径支持差,windows 下建议用 std::Filesystem::u8path()(C++17)包装 UTF-8 字符串。

典型失败场景:

  • std::ofstream 打开失败但没检查 !fout → 静默生成空文件
  • 输入文件被其他程序占用(如 excel 正在编辑 .csv)→ fin 构造失败,is_open() 返回 false
  • 输出路径父目录不存在 → fout 打开失败,不会自动创建目录

真正难调试的是跨平台换行符和 bom:UTF-8 带 BOM 的文本文件头三个字节 0xEF 0xBB 0xBF 会被异或,解密后变成乱码,但你可能以为是密钥错了——其实只是忘了文件开头有 BOM。

text=ZqhQzanResources