0

因此,我一直在尝试使用 python(特别是 pycryptodome)更好地熟悉加密,并且在尝试将字节字符串解码为 ascii 时遇到了一个有趣的问题。请看下面的代码:

from Crypto.Signature import PKCS1_v1_5
from Crypto.Hash import SHA
from Crypto.PublicKey import RSA
message = b'Something secret'

random_gen = Crypto.Random.new().read
print("Type of random_gen: {}".format(type(random_gen)))
private_key = RSA.generate(1024, random_gen) # private key
public_key = private_key.publickey() # public key

signer = PKCS1_v1_5.new(private_key) # signer which uses private key
verifier = PKCS1_v1_5.new(public_key) # verifier which uses public key

h = SHA.new(message) # hash of message
print("Hash: {}".format(h.hexdigest()))

signature = signer.sign(h) # sign hashed version of message
print("Signature type = {}".format(type(signature)))
print("Signature: {}".format(binascii.hexlify(signature).decode('ascii')))

在代码的最后一行,为什么我必须首先hexlify()将签名类型<class 'bytes'>解码为ascii,以便我可以读取签名?为什么如果我这样做:

print("Signature: {}".format(signature.decode('ascii')))

我收到以下错误:

UnicodeDecodeError: 'ascii' codec can't decode byte 0x88 in position 2: ordinal not in range(128)

谢谢您的帮助。

4

1 回答 1

3

signature是一个字节序列:每个元素都是一个介于 0 和 255 之间的整数,如果您尝试直接在 ascii 中对其进行解码,则高于 127 的值将引发异常。

binascii.hexlify从输入返回一个新的字节序列:对于输入的每个字节,在输出中返回两个字节,它们是对应于输入字节的十六进制表示的 ascii 字符代码。因此,输出的每个字节都表示介于'0'and'9'或 between 'a'and之间的 ascii 字符'f'。例如,输入字节128产生两个字符"80",所以两个字节5648(它们是字符'8'和的ascii代码'0')。

因此binascii.hexlify以二进制输入的 ascii 形式生成十六进制表示。decode('ascii')应用后binascii.hexlify不会改变内容,但会产生一个str类型的对象。

在 python 3.5 及更高版本中,您可以简单地使用对象的hex方法bytes来获取str包含其十六进制表示的对象:

signature.hex()
于 2018-07-21T13:27:36.333 回答