问题描述
这在其他语言中可能很简单,但我想不出如何用Solid来实现。
我有一个bytes32
这样的0x05416460deb76d57af601be17e777b93592d8d4d4a4096c57876a91c84f4a712
。
我不希望将字节转换为字符串,而是希望将整个内容表示为字符串,如";0x05416460deb76d57af601be17e777b93592d8d4d4a4096c57876a91c84f4a712";.
如何才能稳妥地做到这一点?
更新:
我需要这样做的原因:基本上我连接到一个Oracle,它离线执行一些工作,最后将文件上传到IPFS。我需要从先知那里获得内容标识符到我的合同中。Oracle只能将bytes32
作为响应发送,因此我将其转换为multihash,并仅将digest
作为bytes32
从Oracle发送到契约。
到目前为止一切顺利,我可以在我的合同中重新创建多哈希。问题是,在此之后,我创建了一个ERC721
(Nft)令牌,并且我必须在元数据中存储对IPFS文件的一些引用,元数据只能是string
格式。这就是我目前被困的地方。
推荐答案
虽然@burt的答案看起来是正确的(虽然没有进行测试),但有一种更有效的方法来解决相同的任务:
function toHex16 (bytes16 data) internal pure returns (bytes32 result) {
result = bytes32 (data) & 0xFFFFFFFFFFFFFFFF000000000000000000000000000000000000000000000000 |
(bytes32 (data) & 0x0000000000000000FFFFFFFFFFFFFFFF00000000000000000000000000000000) >> 64;
result = result & 0xFFFFFFFF000000000000000000000000FFFFFFFF000000000000000000000000 |
(result & 0x00000000FFFFFFFF000000000000000000000000FFFFFFFF0000000000000000) >> 32;
result = result & 0xFFFF000000000000FFFF000000000000FFFF000000000000FFFF000000000000 |
(result & 0x0000FFFF000000000000FFFF000000000000FFFF000000000000FFFF00000000) >> 16;
result = result & 0xFF000000FF000000FF000000FF000000FF000000FF000000FF000000FF000000 |
(result & 0x00FF000000FF000000FF000000FF000000FF000000FF000000FF000000FF0000) >> 8;
result = (result & 0xF000F000F000F000F000F000F000F000F000F000F000F000F000F000F000F000) >> 4 |
(result & 0x0F000F000F000F000F000F000F000F000F000F000F000F000F000F000F000F00) >> 8;
result = bytes32 (0x3030303030303030303030303030303030303030303030303030303030303030 +
uint256 (result) +
(uint256 (result) + 0x0606060606060606060606060606060606060606060606060606060606060606 >> 4 &
0x0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F) * 7);
}
function toHex (bytes32 data) public pure returns (string memory) {
return string (abi.encodePacked ("0x", toHex16 (bytes16 (data)), toHex16 (bytes16 (data << 128))));
}
此代码生成大写输出。对于小写输出,只需将代码中的7更改为39。
说明
我们的想法是使用二进制操作一次处理16个字节。
toHex16
函数将表示为bytes16
值的16字节序列转换为表示为bytes32
值的32位十六进制数字序列。toHex
函数将bytes32
值拆分为两个bytes16
块,通过toHex16
函数将每个块转换为十六进制表示形式,最后使用abi.encodePacked
函数将0x
前缀与转换后的块连接起来。最复杂的部分是toHex16
函数如何工作。让我们逐句解释一下。
第一句:
result = bytes32 (data) & 0xFFFFFFFFFFFFFFFF000000000000000000000000000000000000000000000000 |
(bytes32 (data) & 0x0000000000000000FFFFFFFFFFFFFFFF00000000000000000000000000000000) >> 64;
在这里,我们将输入的最后64位右移64位,基本上是这样做的:
0123456789abcdeffedcba9876543210
\______________/\______________/
| |
| +---------------+
______V_______ ______V_______
/ /
0123456789abcdef0000000000000000fedcba9876543210
第二句:
result = result & 0xFFFFFFFF000000000000000000000000FFFFFFFF000000000000000000000000 |
(result & 0x00000000FFFFFFFF000000000000000000000000FFFFFFFF0000000000000000) >> 32;
这里我们将两个64位区块的最后32位右移32位:
0123456789abcdef0000000000000000fedcba9876543210
\______/\______/ \______/\______/
| | | |
| +-------+ | +-------+
__V___ __V___ __V___ __V___
/ / / /
012345670000000089abcdef00000000fedcba980000000076543210
下一句:
result = result & 0xFFFF000000000000FFFF000000000000FFFF000000000000FFFF000000000000 |
(result & 0x0000FFFF000000000000FFFF000000000000FFFF000000000000FFFF00000000) >> 16;
是否:
012345670000000089abcdef00000000fedcba980000000076543210
\__/\__/ \__/\__/ \__/\__/ \__/\__/
| | | | | | | |
| +---+ | +---+ | +---+ | +---+
V_ V_ V_ V_ V_ V_ V_ V_
/ / / / / / / /
012300004567000089ab0000cdef0000fedc0000ba980000765400003210
和下一个:
result = result & 0xFF000000FF000000FF000000FF000000FF000000FF000000FF000000FF000000 |
(result & 0x00FF000000FF000000FF000000FF000000FF000000FF000000FF000000FF0000) >> 8;
是否:
012300004567000089ab0000cdef0000fedc0000ba980000765400003210
// // // // // // // //
| | | | | | | | | | | | | | | |
| +-+ | +-+ | +-+ | +-+ | +-+ | +-+ | +-+ | +-+
V V V V V V V V V V V V V V V V
/ / / / / / / / / / / / / / / /
01002300450067008900ab00cd00ef00fe00dc00ba00980076005400320010
本系列的最后一句话略有不同:
result = (result & 0xF000F000F000F000F000F000F000F000F000F000F000F000F000F000F000F000) >> 4 |
(result & 0x0F000F000F000F000F000F000F000F000F000F000F000F000F000F000F000F00) >> 8;
将奇数个半字节向右移位4位,将偶数个半字节向右移位8位:
01002300450067008900ab00cd00ef00fe00dc00ba00980076005400320010
| | | | | | | | | | | | | | | |
| | | | | | | | | | | | | | | | | | | | | | | | | | | | | | | |
V V V V V V V V V V V V V V V V V V V V V V V V V V V V V V V V
000102030405060708090a0b0c0d0e0f0f0e0d0c0b0a09080706050403020100
所以初始数据的所有半字节都是按字节分配的。
现在,对于每个字节x
,我们需要执行以下转换:
x` = x < 10 ? '0' + x : 'A' + (x - 10)
让我们稍微重写一下这个公式:
x` = ('0' + x) + (x < 10 ? 0 : 'A' - '0' - 10)
x` = ('0' + x) + (x < 10 ? 0 : 1) * ('A' - '0' - 10)
请注意,(x < 10 ? 0 : 1)
可以计算为((x + 6) >> 4)
,因此我们得到:
x` = ('0' + x) + ((x + 6) >> 4) * ('A' - '0' - 10)
x` = (0x30 + x) + ((x + 0x06) >> 4) * 7
最后声明:
result = bytes32 (0x3030303030303030303030303030303030303030303030303030303030303030 +
uint256 (result) +
(uint256 (result) + 0x0606060606060606060606060606060606060606060606060606060606060606 >> 4 &
0x0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F) * 7);
基本上对每个字节执行上述计算。0x0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F0F
需要右移位后的掩码才能将原始公式中右移位丢弃的位置零。
顺便说一句,像这样的问题最好在https://ethereum.stackexchange.com/
上提问这篇关于固定性:如何将bytes32表示为字符串的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持!