全面掌握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 函数的Gas 消耗

本文通过字符串连接函数为例,来看看如何计算及对比 Solidity 函数的 gas 消耗

Solidity 是以太坊区块链默认使用的智能合约语言,本文讨论了 Solidity 中使用合约和库函数时的 gas 消耗。

背景

在 Datona 实验室的身份合约模板开发过程中,我们希望提供一些有用的错误信息,就需要一些字符串操作,如字符串连接,例如:

function TransferTo(address _address, uint amount) public onlyOwner {
    require(amount <= unallocated, concat("Invalid amount. "
        "Available:", stringOfUint(unallocated)));
    // whatever
}

在Solidity 编译器中,字符串连接是很方便的:

string memory result = string(abi.encodePacked("trust", "worthy"))

但是我们想用一个更有意义的名字来包装它,并将其包含在其他有用的字符串实用函数中(如整数的字符串)

当然,我们想尽可能减少 gas 费用,因为相比普通系统,像 Solidity 这样的区块链语言运行起来是非常昂贵的,毕竟 gas 费用可是一笔不小的钱。

访问方式

为了增加字符串连接功能(它只是一个pure函数,并不能访问状态信息),Solidity 智能合约提供了如下访问方式:

(1) 直接使用编译器提供的方法,如:abi.encodePacked
(2) 继承自基类合约的访问,使用内部(直接)调用
(3) 继承自基类合约的访问,使用外部(间接)调用
(4) 组件合约的访问,使用外部(间接)调用
(5) 库合约的访问,使用内部(直接)调用
(6) 库合约的访问,使用外部(间接)调用

不能通过内部(直接)调用的方式来访问一个组件合约。

参考创建合约,后面章节也有涉及到连接选项的库函数的说明。

接下来的几节会演示不同连接选项的实现。

(2)和(3)继承自基类合约

基类合约提供内部和外部的字符串连接函数(下面归为public),如下所示:

contract Base {
    function Internal(string memory sm0, string memory sm1) 
        internal pure returns (string memory)
    {
        return string(abi.encodePacked(sm0, sm1));
    }
    function External(string memory sm0, string memory sm1)
        public pure returns (string memory)
    {
        return string(abi.encodePacked(sm0, sm1));
    }
}

这里的函数名有点特别,只是为了这篇文章而取的,每一个函数都实现了字符串连接,我们关注的是不同连接选项的性能。

为了能正常使用函数,我们必须实现一个继承合约:

contract MyContract is Base {
    // whatever
}

这些函数可以通过点符号来访问,或者直接省略掉基类合约的名字(你公司的代码标准文档也许有自己的一些规定)

string memory sm = Base.Internal("pass", "word");
    string memory xx = Internal("what", "ever");

(4) 访问组件合约

组件合约被声明为合约的一个组成部分,必须在声明时或在构造函数中创建。


contract Component is Base {...

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

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

0 条评论

请先 登录 后评论