理解以太坊上的事件日志

2020-12-29 11:53:54 浏览数 (1)

  • 译文出自:登链翻译计划[1]
  • 译者:Tiny 熊[2]

序言:先阅读一下 以太坊虚拟机[3],可能会有所帮助,在本文中,我会跳过基础知识直接研究。

在传统编程中,应用程序经常使用日志来捕获和描述特定时刻的情况。这些日志通常用于调试应用程序,检测特定事件或将日志中发生的事情通知查看者。事实证明,在编写智能合约或与智能合约进行交互时,日志也非常有用!那么以太坊是如何做的呢?

以太坊上的日志

EVM 当前有5 个操作码用于触发事件日志:LOG0,LOG1 LOG2 LOG3 和 LOG4。

这些操作码可用于创建“日志记录”。日志记录就是用于描述智能合约中的事件,例如代币转移、所有权变更等。

以太坊黄皮书[4] -拜占庭版本 69351d5(2018-12-10)

每个日志记录都包含“主题(topics)”和“数据”。主题是 32 字节(256 位)的“词”,用于描述事件中发生的事情。不同的操作码(LOG0…LOG4)来描述需要包含在日志记录中的主题数。例如,“ LOG1”包括“一个主题”,而“ LOG4”包括“四个主题”。因此,单个日志记录中可以包含的最大主题数是四个

以太坊日志记录中的主题

日志记录的第一部分由一组主题组成。这些主题用于描述事件。第一个主题通常为事件名称及其参数类型*(uint256,string 等)*签名(keccak256[5]哈希)。一个例外是触发“匿名事件”没有事件签名。由于主题只能容纳 32 个字节的数据,因此无法将数组或字符串等(可能超过 32 个字节)的内容用作主题。而是应将其作为数据包括在日志记录中,而不是作为主题。如果要尝试包含大于 32 个字节的主题,则该主题需要被 hash 计算。因此,仅当你知道原始输入时,才可以知道此哈希表示的内容(译者注:因为 hash 计算不可逆)。

总之,主题应该仅用于需要(压缩)搜索查询(例如:地址)的数据。可以将主题视为事件的索引键,它们都映射到相同的值,接下来将讨论。

以太坊日志记录中的数据

日志记录的第二部分包含额外的数据。主题和数据在一起组成日志记录,主题和数据每自有其优点和缺点。例如,主题是可搜索的,但数据却不能。而数据比主题“便宜得多”。此外,尽管主题最多有 4 个(限制在4 * 32 字节),但数据却没有限制,这意味着它可以包括大量或复杂数据,例如数组或字符串。因此,事件数据(如果有)可以视为

让我们看一些示例,看看主题,数据和日志记录是如何使用的。

触发事件

以下实现了 ERC20 的代币合约,使用了 Transfer 事件:

由于这不是匿名事件,因此第一个主题将包括事件签名:

现在,让我们看一下此 Solidity 事件的参数(from to value):

由于前两个参数声明为indexed,因此被视为主题。最后一个参数没有 indexed ,它将作为数据(而不是单独的主题)。这意味着我们可以进行这样的搜索:查找所有从地址0x0000...(搜索条件)到地址0x0000…(搜索条件)的转账日志,或者是“所有转账到地址0x0000…(搜索条件)的日志”,但没法搜索“转账金额为x(搜索条件)的转账。我们知道了此事件将具有3 个主题,这意味着此日志记录操作将使用LOG3操作码。

现在,我们只需要了解如何包含数据(即最后的参数)即可。LOG3需要 5 个参数:

代码语言:javascript复制
LOG3(memoryStart, memoryLength, topic1, topic2, topic3)

通过以下方式从内存中读取事件数据:

代码语言:javascript复制
memory[memoryStart...(memoryStart   memoryLength)]

幸运的是,像Solidity[6],Vyper[7]或Bamboo[8]这样的高级智能合约程序设计语言将为我们处理将事件数据写入内存的过程,我们可以在触发日志时直接将数据作为参数传递。

检索事件日志

通过使用web3[9] JavaScript 库,可用于与本地或远程以太坊节点进行交互,我们能够订阅新的事件日志:

每当发生新的 SAI 代币转账时,此代码都会通知我们,接收到事件通知,这对很多应用程序都很有用。例如,一旦你在以太坊地址上收到代币,钱包界面就可以提醒你。

日志的 gas 成本

根据黄皮书、日志的基础成本是 375 gas 。另外每个的主题需要额外支付 375 gas 的费用。最后,每个字节的数据需要**8 个 gas **。

这实际上是很便宜!可以计算一下一个 ERC-20 代币转移事件的成本。首先,基本成本为 375 gas 。其次,“转移”事件包含3 个主题,这是另外的 375 _ 3 =1125 gas。最后,我们为所包含的每个数据字节添加8 gas 。由于数据仅包含 ERC-20 转账的数量,最大为32 字节,因此用于记录日志数据所需的最大 gas 量为 8 _ 32 =**256 gas 。这总计要花费1756 gas **的总 gas 成本。作为对比参考,标准的以太币(非代币)转账要花费 21000 gas ,是事件成本的十倍以上了!

如果我们假设 gas 价格为1 gwei,那么操作的总成本将为1756 gwei,相当于0.000001756 ETH。如果以太坊的当前价格在 200 美元左右,那么总计为 $0.0003512。请记住,这是在全球范围内将数据永久存储的费用。

声明:这只是日志记录操作本身的成本。任何以太坊交易至少需要 21000 gas,并且交易的输入数据每字节最多花费 16 gas。通常,要转账和日志记录 ERC-20 代币,费用在 40,000–60,000 gas 。

结论

日志是一种以少量价格将少量数据存储在以太坊区块链上的优雅方法。具体来说,事件日志有助于让其他人知道发生了什么事情,而无需他们单独查询合约。

参考文献

  • Wood,G.(2014)。以太坊:一个安全的去中心化通用交易账本[10]
  • 以太坊基金会 Solidity 文档[11]
  • Web3 文档[12]

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

来源:https://medium.com/mycrypto/understanding-event-logs-on-the-ethereum-blockchain-f4ae7ba50378


参考资料

[1]

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

[2]

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

[3]

以太坊虚拟机: https://medium.com/mycrypto/the-ethereum-virtual-machine-how-does-it-work-9abac2b7c9e

[4]

以太坊黄皮书: https://ethereum.github.io/yellowpaper/paper.pdf

[5]

keccak256: https://en.wikipedia.org/wiki/SHA-3

[6]

Solidity: https://learnblockchain.cn/docs/solidity/

[7]

Vyper: https://github.com/ethereum/vyper

[8]

Bamboo: https://github.com/cornellblockchain/bamboo

[9]

web3: https://learnblockchain.cn/docs/web3.js/

[10]

Wood,G.(2014)。以太坊:一个安全的去中心化通用交易账本: https://ethereum.github.io/yellowpaper/paper.pdf

[11]

以太坊基金会 Solidity 文档: https://solidity.readthedocs.io/en/latest/

[12]

Web3文档: https://learnblockchain.cn/docs/web3.js/

[13]

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

0 人点赞