全面掌握Solidity智能合约开发

2024年09月25日更新 796 人订阅
原价: ¥ 46 限时优惠
专栏简介 跟我学 Solidity :开发环境 跟我学 Solidity:关于变量 跟我学 Solidity : 变量的存储 跟我学 Solidity :引用变量 跟我学 Solidity :函数 跟我学 Solidity :合约的创建和继承 跟我学 Solidity :工厂模式 用Web3.js构建第一个Dapp 跟我学Solidity:事件 Solidity 中 immutable (不可变量)与constant(常量) [译] Solidity 0.6.x更新:继承 解析 Solidity 0.6 新引入的 try/catch 特性 探究新的 Solidity 0.8 版本 探索以太坊合约委托调用(DelegateCall) 停止使用Solidity的transfer() 使用工厂提高智能合约安全性 Solidity 怎样写出最节省Gas的智能合约[译] Solidity 优化 - 编写 O(1) 复杂度的可迭代映射 Solidity 优化 - 控制 gas 成本 Solidity 优化 - 减少智能合约的 gas 消耗的8种方法 Solidity 优化 - 如何维护排序列表 Solidity 优化:打包变量优化 gas 使用 Solidity 瞬态存储操作码 在 Solidity中使用值数组以降低 gas 消耗 Gas 优化:Solidity 中的使用动态值数组 计算Solidity 函数的Gas 消耗 Solidity 技巧:如何减少字节码大小及节省 gas 一些简单的 Gas 优化基础 "Stack Too Deep(堆栈太深)" 解决方案 智能合约Gas 优化的几个技术 合约实践:避免区块Gas限制导致问题 如何缩减合约以规避合约大小限制 Solidity 类特性 无需gas代币和ERC20-Permit还任重而道远 智能合约实现白名单的3个机制 Solidity智能合约安全:防止重入攻击的4种方法 Solidity 十大常见安全问题 [译]更好Solidity合约调试工具: console.log 智能合约开发的最佳实践 - 强烈推荐 全面理解智能合约升级 Solidity可升级代理模式: 透明代理与UUPS代理 使用OpenZeppelin编写可升级的智能合约 实战:调整NFT智能合约,减少70%的铸币Gas成本 Solidity 优化 - 隐藏的 Gas 成本 Gas 技巧:Solidity 中利用位图大幅节省Gas费 Solidity Gas 优化 - 理解不同变量 Gas 差异 关于Solidity 事件,我希望早一点了解到这些 Solidity 编码规范推荐标准 深入了解 Solidity bytes OpenZeppelin Contracts 5.0 版本发布 Solidity Gas优化:高效的智能合约策略 智能合约安全的新最低测试标准:Fuzz / Invariant Test 智能合约的白名单技术 模糊测试利器 - Echidna 简介 智能合约设计模式:代理 离线授权 NFT EIP-4494:ERC721 -Permit

探究新的 Solidity 0.8 版本

如何将你的合约升级到Solidity 0.8?

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

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

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

尤达宝宝发布

0.8新功能和如何使用

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

1. 集成SafeMath

SafeMath Meme

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

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

 transact to Solidity08.test errored: VM error: revert. revert 

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

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

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

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,把它捐给矿工?

更多细节,在这里查看 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: 调用未初始化的内部函数类型的变量。

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

如何迁移到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

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


本翻译由 Cell Network 赞助支持。

点赞 3
收藏 2
分享
本文参与登链社区写作激励计划 ,好文好收益,欢迎正在阅读的你也加入。

0 条评论

请先 登录 后评论