全面掌握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

一些简单的 Gas 优化基础

  • 影无双
  • 发布于 2022-03-24 16:00
  • 阅读 6003

普通开发者需要了解的优化 Solidity 合约的基础

普通开发者需要了解的优化 Solidity 合约的基础

编写智能合约是很难的。不仅是要确保代码没有漏洞,而且你的编写方式还会影响到用户与它交互时的开销。

当你在编译智能合约时,每一行 Solidity 代码都会转换为一系列的操作(操作码),这些操作都有对应的 gas 消耗。你的目标就是要让你的程序使用尽可能少的操作码(或者用更便宜的)。

当然,这些都很复杂,所以,我们要慢慢来。与其陷入操作码兔子洞,不如尝试一些可以直接应用到合约里的简单优化。

升级 Solidity 版本

合约中,Solidity 版本是在文件最顶部定义的,像这样:

pragma solidity ^0.8.0;

在这里,^0.8.0意思是合约使用0.8.x系列最新可用的Solidity版本。

更新的 Solidity 版本有时会在修复bug和安全补丁时就优化了 gas ,所以,升级到最新版本不仅会让你的代码更安全,通常也会更便宜。

要捕获最近大多数优化,请确保你的版本在0.8.4以上:

pragma solidity ^0.8.4;

放弃 Counters.sol

如果你的 NFT 项目或者代币正在使用 OpenZeppelin 合约,很可能你正在用 OZ 的Counters.sol库。

在较新的 Solidity 版本(0.8的更高版本),这个库并不是很有用,用常规整数替代它可以节省一些gas:

contract TestCounters {
-   using Counters for Counters.Counter;
-   Counters.Counter private _tokenIds;

+   uint256 private _tokenId;

    function mint() public {
-       _tokenIds.increment();
-       uint256 tokenId = _tokenIds.current();

+       uint256 tokenId = _tokenId++;
    }
}

标记不可变变量

无论是代币的小数位数,USDC 的地址,还是支付账户,有时我们并不打算更改合约变量。此时,将它们标记为常量(如果你在代码中编写它们)或者不可变量(如果你计划之后给它们赋值,比如,通过构造函数)可以降低访问这些值时的开销:

contract TestImmutable {
    uint256 internal constant DECIMALS = 18;
    address public immutable currencyToken;

    constructor(address _currencyToken) {
        currencyToken = _currencyToken;
    }
}

unchecked {}

从 Solidity 0.8开始,所有数学运算都包括溢出检查。这是很棒的(替换了 SafeMath 库,如果你还在用可以丢弃了),但是它需要额外的 gas 开销,所以我们想在不必要的时候绕开它。

溢出检查的意义在于帮你检查是否存在从 0 减...

剩余50%的内容订阅专栏后可查看

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

0 条评论

请先 登录 后评论