a1 = b'\x01\x02\x41'
\x41
是A
当我打印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
答案 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)
并将生成的二进制数据写出到以二进制模式打开的文件中。