如何将字节转换为整个十六进制字符串?

时间:2018-05-22 15:04:06

标签: python redis byte

a1 = b'\x01\x02\x41'

\x41A

当我打印str(a1)时,我得到了

b'\x01\x02A'

我怎么能得到原始的十六进制字符串?

我想得到:

\x01\x02\x41

我将此字符串用于redis-cli --pipe

我不希望python为我打印转换后的字符串。

我怎么能这样做?

现在我喜欢这个:

def convert(i:bytes):
    result = list()
    for j in i:
        result.append('\\x%s' % bytes([j]).hex())
    return ''.join(result)

太难看了。 还有更好的方法吗?

顺便说一句。

这是我的redis-cli --pipe脚本:

SET "\xda\xb8\xe6\xc0\x79\x62\xea\x06\xde\x9e\xb0\x4e\x68\xde\x0b\x62" "\x00\x42\xdc\x06\x03\x00\x00\x00"
SET "\x68\xfa\xfc\x03\x36\x99\xb0\x56\xb4\x00\xaf\x61\x42\xe0\x30\x42" "\x01\x42\xdc\x06\x03\x00\x00\x00"
SET "\x40\x98\xb9\x8f\xd8\x4e\x2e\x32\x40\x09\xca\xa4\x55\x52\xde\x7f" "\x02\x42\xdc\x06\x03\x00\x00\x00"
SET "\xd0\x75\xdf\x46\x36\xd0\xb4\xed\xcc\xed\xd6\x27\xf7\xd9\xc5\xa5" "\x03\x42\xdc\x06\x03\x00\x00\x00"

我这样使用它:

cat sample.redis | redis-cli --pipe

1 个答案:

答案 0 :(得分:0)

当迭代bytes值时,你得到整数;这些可以简单地转换为十六进制表示法:

def convert(value: bytes):
    return ''.join([f'\\x{b:02x}' for b in value])

请注意,这会生成一个包含文字反斜杠,x字符和十六进制数字字符的字符串。它不再是bytes值。

演示:

>>> print(convert(b'\x01\x02\x41'))
\x01\x02\x41

只是为了确保您无需担心bytes。当字节值恰好是可打印的ASCII代码点时,repr()对象的bytes表示将始终使用ASCII字符。这并不意味着价值会发生变化。 b'\x01\x02\x41'等于b'\x01\x02A'。 Redis协议对\x<HH>转义序列没有任何了解,因此请勿尝试通过线路发送上述字符串。

您生成的转义序列是bash shell string sequences,就像Python字符串一样,您不必使用转义。与在Python中一样,对于Bash,字符串"\x01\x02A""\x01\x02\x41"具有相同的值。只有在传递命令行中的键和值字符串时才有意义,而不是在管道传输到redis-cli的文本文件中。

此外,请注意redis-cli --pipe命令采用原始Redis协议输入,而不是Redis命令语法,请参阅Redis Mass Insertion documentation。该协议使用\xhh序列,因为它不使用shell表示法。

而是使用以下函数生成原始SET命令:

def raw_protocol(cmd: str, *args: bytes):
    return b'\r\n'.join([
        f'*{len(args) + 1}\r\n${len(cmd)}\r\n{cmd}'.encode(),
        *(bv for a in args for bv in (b'$%d' % len(a), a)),
        b''
    ])

对于SET命令,使用raw_protocol('SET', keybytes, valuebytes)并将生成的二进制数据写出到以二进制模式打开的文件中。