5

我需要在solidity智能合约中存储这种类型的值0xff00000x00ff08(十六进制颜色表示),并能够在合约中将其转换为具有相同文本字符的字符串"ff0000"。我打算在 RSK 上部署这个智能合约。

我的想法是将这些值存储在一个bytes3或简单的uint变量中,并有一个纯函数转换bytes3或转换uint为相应的字符串。我找到了一个可以完成这项工作并在solidity 0.4.9上工作的函数

pragma solidity 0.4.9;

contract UintToString {
    function uint2hexstr(uint i) public constant returns (string) {
        if (i == 0) return "0";
        uint j = i;
        uint length;
        while (j != 0) {
            length++;
            j = j >> 4;
        }
        uint mask = 15;
        bytes memory bstr = new bytes(length);
        uint k = length - 1;
        while (i != 0){
            uint curr = (i & mask);
            bstr[k--] = curr > 9 ? byte(55 + curr ) : byte(48 + curr); // 55 = 65 - 10
            i = i >> 4;
        }
        return string(bstr);
    }
}

但我需要更新的编译器版本(至少 0.8.0)。上述功能不适用于较新的版本。

在 Solidity >=0.8.0 中有效的转换bytes或转换为十六进制字符串 (1->'1',f->'f') 的方法是什么?uint

4

1 回答 1

3

以下编译,并使用 solc 0.8.7 测试它与您的原始版本相同,具有以下修改:

  • constant-->pure
  • returns (string)-->returns (string memory)
  • byte(...)--> bytes1(uint8(...))

上述更改克服了原始函数中的所有编译时差异。...但是仍然存在导致此功能恢复的运行时错误:

在调试时,该行在bstr[k--] = curr > 9 ?其所在循环的最后一次迭代中触发了还原。这是因为 while 循环的设置k0在其最后一次迭代中。

虽然识别很棘手,但修复很简单:将减量运算符从后缀更改为前缀 - bstr[--k] = curr > 9 ?

在旁边:

问:为什么在 solc 0.4.9 中编译时不会恢复,但在 solc 0.8.7 中编译相同的代码时会恢复?

答:solc 0.8.0 引入了一项重大更改,其中编译器插入了 uint 上溢和下溢检查。在此之前,需要使用SafeMath或类似的方法来完成相同的操作。 请参阅 solc 0.8 发行说明的“语义的静默变化”部分

pragma solidity >=0.8;

contract TypeConversion {
    function uint2hexstr(uint i) public pure returns (string memory) {
        if (i == 0) return "0";
        uint j = i;
        uint length;
        while (j != 0) {
            length++;
            j = j >> 4;
        }
        uint mask = 15;
        bytes memory bstr = new bytes(length);
        uint k = length;
        while (i != 0) {
            uint curr = (i & mask);
            bstr[--k] = curr > 9 ?
                bytes1(uint8(55 + curr)) :
                bytes1(uint8(48 + curr)); // 55 = 65 - 10
            i = i >> 4;
        }
        return string(bstr);
    }
}

于 2021-09-23T14:52:49.827 回答