Python AES 加密解密后为空字符串问题的解决

python aes 加密解密后为空字符串问题的解决

本文旨在解决在使用 Python 的 Crypto 库进行 AES 加密和解密时,解密后得到空字符串的问题。通过分析常见原因和提供修复后的代码示例,帮助开发者正确实现 AES 加密解密功能,确保数据的安全传输和存储。

AES(Advanced Encryption Standard)是一种广泛使用的对称加密算法,在数据安全领域扮演着重要角色。 然而,在使用 Python 的 Crypto 库实现 AES 加密和解密时,开发者可能会遇到解密后得到空字符串的问题。这通常是由于密钥处理不当造成的。以下将详细介绍如何正确处理密钥,并提供完整的代码示例。

问题分析

在提供的代码中,AESCipher 类的 __init__ 方法中,当用户提供密钥时,会对密钥进行哈希处理:

self.key = hashlib.sha256(key.encode()).digest()

而 get_key 方法返回的是密钥的 Base64 编码

立即学习“Python免费学习笔记(深入)”;

return b64encode(self.key).decode("utf-8")

这意味着,当从文件中读取密钥并用于解密时,实际上使用的是哈希后的密钥的 Base64 编码,而不是原始密钥。 这会导致解密失败,从而得到空字符串。

解决方案

正确的做法是,在 AESCipher 的构造函数中,如果提供了密钥,则应该对其进行 Base64 解码,而不是进行哈希处理。修改后的 __init__ 方法如下:

class AESCipher(object):    def __init__(self, key=None):        # Initialize the AESCipher object with a key,        # defaulting to a randomly generated key        self.block_size = AES.block_size        if key:            self.key = b64decode(key.encode())        else:            self.key = Random.new().read(self.block_size)

完整代码示例

以下是修改后的完整代码示例:

import hashlibfrom Crypto.Cipher import AESfrom Crypto import Randomfrom base64 import b64encode, b64decodeclass AESCipher(object):    def __init__(self, key=None):        # Initialize the AESCipher object with a key,        # defaulting to a randomly generated key        self.block_size = AES.block_size        if key:            self.key = b64decode(key.encode())        else:            self.key = Random.new().read(self.block_size)    def encrypt(self, plain_text):        # Encrypt the provided plaintext using AES in CBC mode        plain_text = self.__pad(plain_text)        iv = Random.new().read(self.block_size)        cipher = AES.new(self.key, AES.MODE_CBC, iv)        encrypted_text = cipher.encrypt(plain_text)        # Combine IV and encrypted text, then base64 encode for safe representation        return b64encode(iv + encrypted_text).decode("utf-8")    def decrypt(self, encrypted_text):        # Decrypt the provided ciphertext using AES in CBC mode        encrypted_text = b64decode(encrypted_text)        iv = encrypted_text[:self.block_size]        cipher = AES.new(self.key, AES.MODE_CBC, iv)        plain_text = cipher.decrypt(encrypted_text[self.block_size:])        return self.__unpad(plain_text)    def get_key(self):        # Get the base64 encoded representation of the key        return b64encode(self.key).decode("utf-8")    def __pad(self, plain_text):        # Add PKCS7 padding to the plaintext        number_of_bytes_to_pad = self.block_size - len(plain_text) % self.block_size        padding_bytes = bytes([number_of_bytes_to_pad] * number_of_bytes_to_pad)        padded_plain_text = plain_text.encode() + padding_bytes        return padded_plain_text    @staticmethod    def __unpad(plain_text):        # Remove PKCS7 padding from the plaintext        last_byte = plain_text[-1]        return plain_text[:-last_byte] if isinstance(last_byte, int) else plain_textdef save_to_notepad(text, key, filename):    # Save encrypted text and key to a file    with open(filename, 'w') as file:        file.write(f"Key: {key}nEncrypted text: {text}")    print(f"Text and key saved to {filename}")def encrypt_and_save():    # Take user input, encrypt, and save to a file    user_input = ""    while not user_input:        user_input = input("Enter the plaintext: ")    aes_cipher = AESCipher()  # Randomly generated key    encrypted_text = aes_cipher.encrypt(user_input)    key = aes_cipher.get_key()    filename = input("Enter the filename (including .txt extension): ")    save_to_notepad(encrypted_text, key, filename)def decrypt_from_file():    # Decrypt encrypted text from a file using a key    filename = input("Enter the filename to decrypt (including .txt extension): ")    with open(filename, 'r') as file:        lines = file.readlines()        key = lines[0].split(":")[1].strip()        encrypted_text = lines[1].split(":")[1].strip()    aes_cipher = AESCipher(key)    decrypted_bytes = aes_cipher.decrypt(encrypted_text)    # Decoding only if the decrypted bytes are not empty    decrypted_text = decrypted_bytes.decode("utf-8") if decrypted_bytes else ""    print("Decrypted Text:", decrypted_text)def encrypt_and_decrypt_in_command_line():    # Encrypt and then decrypt user input in the command line    user_input = ""    while not user_input:        user_input = input("Enter the plaintext: ")    aes_cipher = AESCipher()    encrypted_text = aes_cipher.encrypt(user_input)    key = aes_cipher.get_key()    print("Key:", key)    print("Encrypted Text:", encrypted_text)    decrypted_bytes = aes_cipher.decrypt(encrypted_text)    decrypted_text = decrypted_bytes.decode("utf-8") if decrypted_bytes else ""    print("Decrypted Text:", decrypted_text)# Menu Interfacewhile True:    print("nMenu:")    print("1. Encrypt and save to file")    print("2. Decrypt from file")    print("3. Encrypt and decrypt in command line")    print("4. Exit")    choice = input("Enter your choice (1, 2, 3, or 4): ")    if choice == '1':        encrypt_and_save()    elif choice == '2':        decrypt_from_file()    elif choice == '3':        encrypt_and_decrypt_in_command_line()    elif choice == '4':        print("Exiting the program. Goodbye!")        break    else:        print("Invalid choice. Please enter 1, 2, 3, or 4.")

注意事项

密钥管理: 密钥的安全至关重要。不要将密钥硬编码到代码中,而是使用安全的方式存储和管理密钥。编码: 确保在加密和解密过程中使用一致的编码方式,例如 UTF-8。Padding: AES 需要对明文进行填充,以确保其长度是块大小的倍数。 PKCS7 是一种常用的填充方案。IV (Initialization Vector): 在使用 CBC 模式时,需要使用一个随机的 IV。 IV 不需要保密,但必须在加密和解密过程中使用相同的 IV。

总结

通过正确处理密钥,可以避免 AES 解密后得到空字符串的问题。 在实际应用中,还需要注意密钥管理、编码和填充等问题,以确保数据的安全。 本文提供的代码示例可以作为 AES 加密解密的基础,开发者可以根据实际需求进行修改和扩展。

以上就是Python AES 加密解密后为空字符串问题的解决的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1375191.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月14日 14:47:59
下一篇 2025年12月14日 14:48:08

相关推荐

  • Python AES 加密解密后文本为空的解决方案

    本文针对 Python 中使用 Crypto 库进行 AES 加密解密时出现解密后文本为空的问题,提供了一种解决方案。通过分析代码,指出问题在于密钥处理方式,并提供修正后的代码示例,确保加密解密流程的正确性。同时,本文还包含完整的加密解密示例代码,方便读者理解和应用。 在使用 Python 的 Cr…

    好文分享 2025年12月14日
    000
  • 创建 TensorFlow 自定义优化器:获取梯度和变量向量

    本文档旨在指导开发者如何在 TensorFlow 中创建自定义优化器。我们将重点介绍如何获取每次迭代中的梯度和变量向量,并正确地更新这些值。通过继承 tf.keras.optimizers.Optimizer 类,并重写关键方法,开发者可以灵活地实现自己的优化算法,从而更好地控制模型的训练过程。本文…

    2025年12月14日
    000
  • 在 TensorFlow 中构建自定义优化器

    本文档旨在指导开发者如何在 TensorFlow 中创建自定义优化算法。我们将深入探讨如何获取每次迭代的当前点向量 x 和梯度向量 g,以及如何更新 x 并将更新后的值设置回模型。通过一个具体的示例,我们将展示如何修改梯度形状以适应自定义优化算法的需求,并提供构建和应用自定义优化器的完整流程。 自定…

    2025年12月14日
    000
  • Python列表中有哪些索引

    Python列表支持四种索引方式:1. 正数索引从0开始访问元素,如my_list[0]获取第一个元素;2. 负数索引从-1起从末尾反向访问,如my_list[-1]获取最后一个元素;3. 切片索引用[start:end:step]获取子列表,支持步长与反转;4. 动态索引通过index()方法查找…

    2025年12月14日
    000
  • TensorFlow自定义优化器教程:深入理解梯度操作

    本文旨在指导开发者如何在TensorFlow中创建自定义优化器,重点讲解如何获取每次迭代的当前点向量和梯度向量,并进行更新。通过实例代码,详细解释了梯度扁平化处理的重要性,以及如何在自定义优化器中正确更新模型参数,从而实现对神经网络优化算法的灵活控制。 在TensorFlow中,自定义优化器能够让我…

    2025年12月14日
    000
  • 高效处理 Python 异步操作中的异常

    本文旨在提供一种在 Python 的 asyncio 框架下,高效处理异步操作中异常的方法。重点在于如何在单个任务发生异常时,避免影响其他并发任务的执行,从而保证程序的健壮性和稳定性。我们将通过代码示例展示如何在异步函数内部进行异常处理,确保即使出现错误,程序也能继续执行。 在 Python 中使用…

    2025年12月14日
    000
  • Python异步操作中的高效错误处理

    本文旨在提供在Python中使用asyncio进行异步操作时,如何高效处理错误的实用指南。通过将错误处理逻辑嵌入到每个独立的异步任务中,可以确保即使某个任务失败,其他任务也能继续执行,从而提高程序的整体健壮性和可靠性。本文将提供示例代码,演示如何在asyncio中实现这种错误处理策略,并讨论相关的注…

    2025年12月14日
    000
  • 通过值搜索Python字典并返回相关值

    本文旨在提供一种通过Python字典的值来查找相关信息的方法。我们将探讨如何遍历字典,检查目标值是否存在,并返回与该值相关的其他值。通过示例代码,你将学会如何根据元素名称、符号、原子序数或原子量来查找化学元素的相关信息。同时,我们也会讨论如何保持返回值的顺序,以及在实际应用中需要注意的事项。 在许多…

    2025年12月14日
    000
  • Python字典高效过滤:创建满足特定条件的新字典

    本文将介绍如何高效地根据特定条件过滤Python字典,并创建两个新的字典。原始字典中的每个键值对将根据值中是否包含特定字符串而被分配到两个新字典中的一个。我们将使用 items() 方法遍历字典,并结合 any() 函数进行条件判断,以实现简洁高效的过滤。 使用 items() 方法和 any() …

    2025年12月14日
    000
  • 使用 Python 过滤字典创建新字典的实用指南

    本文旨在指导读者如何根据特定条件,高效地从一个Python字典中创建两个新的字典。通过迭代原始字典的键值对,并结合any()函数进行条件判断,我们可以将符合条件的键值对分别添加到两个目标字典中,从而实现字典的过滤和拆分。本文提供清晰的代码示例和详细的解释,帮助读者理解和应用这一技术。 在Python…

    2025年12月14日
    000
  • Python中os模块的功能介绍

    os模块提供跨平台系统交互功能,支持文件目录操作(如创建、删除、重命名)、路径处理(拼接、判断存在性)、环境变量管理及系统命令执行,常用于自动化脚本,需注意权限与路径兼容性问题。 Python的os模块提供了与操作系统交互的接口,允许开发者执行文件和目录操作、管理进程、处理路径等。它是一个跨平台模块…

    2025年12月14日
    000
  • python函数形参如何设置默认值

    在Python中,函数形参可通过等号设置默认值,如def greet(name, prefix=”Hello”),调用时若未传参则使用默认值,且默认参数需位于非默认参数之后,避免使用可变对象作为默认值,因默认值在定义时即确定,正确做法是用None判断并初始化,从而提升函数灵活…

    2025年12月14日
    000
  • python字典遍历所有的键值对

    最常用的是使用items()方法遍历键值对,还可通过keys()遍历键、values()遍历值,或直接遍历字典获取键。 在Python中,遍历字典的所有键值对有几种常用方法。最直接的方式是使用 items() 方法,它返回一个包含字典中所有键值对的可迭代对象。 使用 items() 遍历键值对 通过…

    2025年12月14日
    000
  • python控制语句的两大分类

    Python控制语句分为条件和循环两类。条件语句包括if、if-else、if-elif-else,用于根据条件真假执行不同代码块;循环语句包括for和while循环,用于重复执行代码,其中for遍历序列,while在条件为真时持续执行,循环中可用break退出、continue跳过当前次、pass…

    2025年12月14日
    000
  • python如何为函数和模块起别名

    在Python中,as关键字可用于为模块或函数设置别名。例如import numpy as np后可用np调用numpy功能;from math import sqrt as square_root后可用square_root调用sqrt函数。别名常用于缩短长模块名、避免命名冲突、提升可读性,使代码…

    2025年12月14日
    000
  • 在 GitHub 上展示 Python 项目的代码覆盖率

    本文将指导你如何在 GitHub 上配置 Python 项目的代码覆盖率,以便每次推送时都能自动生成覆盖率报告。我们将使用 pytest-cov 工具来生成覆盖率数据,并将其上传到 Codecov 等平台进行可视化展示,从而帮助你更好地了解代码的测试情况。 使用 pytest-cov 生成覆盖率报告…

    2025年12月14日
    000
  • 通过值查找Python字典并返回相关信息

    本文介绍了如何使用Python在一个包含元素信息的字典中,通过元素名称、符号、原子序数或原子量等值来查找并返回与该元素相关的其他信息。文章提供了详细的代码示例,并针对字典值存储方式可能导致的问题提出了解决方案,旨在帮助读者高效地实现基于值的字典搜索功能。 在化学应用或其他需要快速检索元素信息的场景中…

    2025年12月14日
    000
  • 基于值的字典搜索:Python 教程

    本文档旨在指导读者如何使用 Python 搜索字典中的值,并返回与该值相关联的其他值。我们将提供代码示例,演示如何遍历字典并查找特定值,然后提取与该值关联的其他信息。此外,我们还会讨论如何处理值的顺序问题,并提供一些优化建议。 字典值搜索方法 在 Python 中,字典是一种非常有用的数据结构,它允…

    2025年12月14日
    000
  • python字典的应用场景

    字典通过键值对实现高效查找,适用于数据映射、计数统计、缓存记忆化和结构化数据表示,具有O(1)平均时间复杂度,广泛用于配置管理、频率统计、递归优化及Web数据处理。 Python字典是一种非常灵活且高效的数据结构,适用于多种实际场景。它的核心特点是通过键(key)快速查找对应的值(value),具有…

    2025年12月14日
    000
  • python import的本质探究

    import不仅加载代码,还通过sys.modules缓存模块,确保唯一性;它按sys.path搜索路径查找模块,执行顶层代码并创建命名空间;模块级变量共享,支持相对导入与包结构,需注意循环导入和可变对象副作用。 import在Python中不只是加载代码,它背后有一套完整的机制来确保模块的正确加载…

    2025年12月14日
    000

发表回复

登录后才能评论
关注微信