前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >探究新的 Solidity 0.8 版本

探究新的 Solidity 0.8 版本

作者头像
Tiny熊
发布2021-02-25 11:36:54
1.5K0
发布2021-02-25 11:36:54
举报
文章被收录于专栏:深入浅出区块链技术

  • 译文出自:登链翻译计划[1]
  • 译者:翻译小组[2]
  • 校对:Tiny 熊[3]

我们离 Solidity 1.0 的发布越来越近了(当然除非 0.9 之后是 0.10)。Solidity0.8[4]在0.7 发布[5]之后仅 5 个月就发布了!

登链社区正在同步翻译 Solidity 0.8 的文档[6]

今天我们就来探讨一下如何把合约迁移升级到 0.8 版本... ...

尤达宝宝发布

0.8 新功能和如何使用

我们来看看两个大的新功能:集成的 SafeMath 和新的错误处理。

1. 集成 SafeMath

SafeMath Meme

没错,你不需要再导入 Openzeppelin SafeMath[7]了。最重要的是,你不需要做任何事情就可以激活 Solidity 集成的 SafeMath。只要写上a+b,就会在溢出时自动回退交易。

你可能会在 Remix 等工具中看到如下错误提示,因为 0.8 还没有完全支持。比如溢出还没有给出确切的原因。

代码语言:javascript
复制
 transact to Solidity08.test errored: VM error: revert. revert

但这种情况在未来应该会有所改变。

如果你确实就希望代码能够溢出呢?还是非常在意 Gas 费了?

则通过 unchecked形式包装语句来停用 SafeMath:

代码语言:javascript
复制
contract Solidity08 {
    function test() external pure returns(uint256) {
        // 默认使用 SafeMath ,此时会回退
        uint256 x = 0;
        x--;

        return x;
    }
}

contract Solidity08 {
    function test() external pure returns(uint256) {
        // 不使用SafeMath,返回 type(uint256).max

        uint256 x = 0;
        unchecked { x--; }

        return x;
    }
}

2. 无效的操作码被还原取代

到目前为止,某些操作会导致 INVALID操作码的执行。这个操作码的问题是,它消耗掉所有剩余的 Gas。这显然是不好的且没必要。为什么要浪费 Gas,把它捐给矿工?

更多细节,在这里[8]查看 revertassert的区别。

现在 Solidity 使用revert操作码。为了区分常规 revert 和系统内部错误(panic),Solidity 在返回数据前加上一个标识符:

  • 常规 Revert 错误keccak256(Error(string))的前四个字节开始,等于0x08c379a0
  • **系统内部错误(Panic)**以keccak256(Panic(uint256))的前四个字节开始,等于0x4e487b71

Panic 有一个额外的错误标识。目前可用的 Panic 有:

  • 0x01: 使用 asset
  • 0x11: SafeMath 的溢出。
  • 0x12: 除以 0。
  • 0x21: 转换为不存在的枚举类型。
  • 0x22: 存储字节数组编码错误。
  • 0x31: 在一个空数组上pop()
  • 0x32: 索引超长度异常。
  • 0x41: 分配过多的内存或创建过大的数组。
  • 0x51: 调用未初始化的内部函数类型的变量。

更多细节请参见文档中新的错误处理部分这里[9]

如何迁移到 Solidity 0.8

在大多数情况下,迁移应该是非常直接的。只有在一些情况下,你做奇怪的类型转换可能会变得更加困难。

你必须为迁移做出的改变包括:

  • ABIEncoderV2现在是默认自动激活。从 0.6 开始,Encoder 就不再是实验性的了,只是因为遗留的原因,保留了 pragma experimental这个名字。现在你不需要再加这行了。
  • 移除任何 Openzeppelin SafeMath,你不再需要它了。
  • 可能需要进行一些类型转换。
    • msg.sendertx.origin默认不属于 payable 类型。将 msg.sender.transfer改为 payable(msg.sender).transfer
    • 只有在符合给定类型的情况下,才允许类型转换,所以uint256(-1)将不再工作。使用type(uint256).max代替。
    • 当多次改变符号时,类型转换在某些情况下会受到限制,因为类型转换的顺序可能会对结果产生影响。你现在会看到一个类似 TypeError 的错误。不允许从 int256bytes32进行显示的类型转换,得先手动转换为 uint256。
  • 修饰组合
    • myContract.functionCall{gas: 10000}{value: 1 ether }()改为:
    • myContract.functionCall{gas: 10000, value: 1 ether }()
  • x**y**z改为(x**y)**z,因为默认的执行顺序改变了。
  • byte类型改为 byte1

我省略了一些细节,关于完整的变更日志和所有变更的细节,请查看文档这里[10]


本翻译由 Cell Network[11] 赞助支持。

来源:https://soliditydeveloper.com/solidity-0.8

参考资料

[1]

登链翻译计划: https://github.com/lbc-team/Pioneer

[2]

翻译小组: https://learnblockchain.cn/people/412

[3]

Tiny 熊: https://learnblockchain.cn/people/15

[4]

Solidity0.8: https://blog.soliditylang.org/2020/12/16/solidity-v0.8.0-release-announcement/

[5]

0.7发布: https://blog.soliditylang.org/2020/07/28/solidity-v0.7.0-release-announcement/

[6]

Solidity 0.8 的文档: https://learnblockchain.cn/docs/solidity/

[7]

SafeMath: https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/contracts/math/SafeMath.sol

[8]

这里: https://medium.com/blockchannel/the-use-of-revert-assert-and-require-in-solidity-and-the-new-revert-opcode-in-the-evm-1a3a7990e06e

[9]

这里: https://docs.soliditylang.org/en/latest/control-structures.html#panic-via-assert-and-error-via-require

[10]

这里: https://docs.soliditylang.org/en/latest/080-breaking-changes.html

[11]

Cell Network: https://www.cellnetwork.io/?utm_souce=learnblockchain

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2021-02-06,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 深入浅出区块链技术 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 0.8 新功能和如何使用
    • 1. 集成 SafeMath
      • 2. 无效的操作码被还原取代
      • 如何迁移到 Solidity 0.8
        • 参考资料
        相关产品与服务
        区块链
        云链聚未来,协同无边界。腾讯云区块链作为中国领先的区块链服务平台和技术提供商,致力于构建技术、数据、价值、产业互联互通的区块链基础设施,引领区块链底层技术及行业应用创新,助力传统产业转型升级,推动实体经济与数字经济深度融合。
        领券
        问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档