我有一份简单的合同,如下所示:
contract storage {
bytes32 data;
function f(uint256 _a, uint256 _b) public {
bytes32 hash = keccak256(abi.encode(_a, _b));
data = hash;
}
}我想理解为什么称函数f值在44k左右,因为我读到SSTORE最多要花费22k气体。煤气费从何而来?谢谢。
发布于 2022-10-11 02:28:09
在这里,您有一些最相关的操作在气体方面。
让我们假设您将一个事务发送给函数f,分别作为_a和_b传递1和2。
这意味着您将有以下输入:
0x13d1aa2e00000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000002
功能选择器:0x13d1a2e
参数0000000000000000000000000000000000000000000000000000000000000001:_a
参数0000000000000000000000000000000000000000000000000000000000000002:_b
如果您查看白花黄纸 (第27页),您将看到:您将为以下操作支付多少费用:
Gtxdatazero 4(气体单位)支付一个事务的数据或代码的零字节。Gtxdatanonzero 16 (气体单位)支付事务的每一个非零字节的数据或代码。
0x13d1aa2e (函数选择器)4非零字节=> 4* 16 (气体单位)= 64
参数_a: 31 0字节+1非零字节=> 31 *4+1* 16 = 140
参数_b: 31 0字节+1非零字节=> 31 *4+1* 16 = 140
Gkeccak256 30为每次KECCAK256手术付费。Gkeccak256word 6为KECCAK256操作输入数据的每个单词(四舍五入)付费
keccak256(abi.encode(_a,_b));
abi.encode(_a,_b)的结果是:
0x00000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000002
意思是两个单词作为keccak256的输入
每次KECCAK256行动付费30 => 30
为KECCAK256操作=> 6*2= 12输入数据的每个单词(四舍五入)付费
当存储值从零Gcoldsload 2100成本设置为非零时(首次在事务中访问状态变量),Gsset 20000支付了SSTORE操作。
Gtransaction 21000支付了每笔交易
此外,您还可以使用混合方法调试它,并检查执行的每个操作码和这里,您可以查看它的成本。

发布于 2022-10-11 01:39:19
最典型的原因是数据的存储。如果将以前的非set变量的值更改为set变量,则事务成本超过重置相同变量的值。
https://ethereum.stackexchange.com/questions/137256
复制相似问题