您观察到的行为的原因
abi.encodePacked(myBytes3)
产生 3 个字节,因为每 2 个十六进制字符产生 1 个字节。所以当你把它包装进去时,string(..)
你会得到一个每 1 个字节有 1 个字符的字符串,在这种情况下,这会产生一个有 3 个字符的字符串。对于某些输入,这会产生一个可以呈现为人类可读字符串的字符串。在其他情况下,它会导致字符串不能,因此会出现解析错误。
解决方案
在这种情况下您不能使用abi.encodePacked(..)
,因为“打包”与您尝试完成的相反bytes3
- 将 a 呈现为十六进制string
- 一种人类可读的 ASCII 编码 - 您需要做一些类似于
我的回答你之前的问题,你使用位移位和位掩码的组合来每半字节提取一个字符。
代码
将单个uint8
转换为 ASCII 字符的实用函数,该 ASCII 字符将用十六进制表示它:
function uint8tohexchar(uint8 i) public pure returns (uint8) {
return (i > 9) ?
(i + 87) : // ascii a-f
(i + 48); // ascii 0-9
}
uint24
将 a 转换为 6 个字符的十六进制的函数string
。
请注意,由于这次您不是在寻找“通用”解决方案,而是特定于 的解决方案,bytes3
因此不需要任何循环,而是一组连续的语句就可以了。可能也节省了一些汽油。(有待确认!)
function uint24tohexstr(uint24 i) public pure returns (string memory) {
bytes memory o = new bytes(6);
uint24 mask = 0x00000f;
o[5] = bytes1(uint8tohexchar(uint8(i & mask)));
i = i >> 4;
o[4] = bytes1(uint8tohexchar(uint8(i & mask)));
i = i >> 4;
o[3] = bytes1(uint8tohexchar(uint8(i & mask)));
i = i >> 4;
o[2] = bytes1(uint8tohexchar(uint8(i & mask)));
i = i >> 4;
o[1] = bytes1(uint8tohexchar(uint8(i & mask)));
i = i >> 4;
o[0] = bytes1(uint8tohexchar(uint8(i & mask)));
return string(o);
}
最后,您可能需要一个明确接受bytes3
作为参数的函数。
请注意,这纯粹是为了方便,因为它所做的只是类型转换。和可以简单地认为是“24 位”以不同的方式查看/解释bytes3
。uint24
function bytes3tohexstr(bytes3 i) public pure returns (string memory) {
uint24 n = uint24(i);
return uint24tohexstr(n);
}