彻底理解solidity中的事件

2023-02-24 10:39:12 浏览数 (1)

在我之前的几篇关于智能合约的文章中,都有提到事件的用法,比如:

代码语言:javascript复制
event HighestBidIncreased(address bidder, uint amount);
event AuctionEnded(address winner, uint amount);

这里定义了两个事件,分别表示最高竞价更新了和拍卖结束了。

然后在需要的位置,调用事件,比如:

代码语言:javascript复制
function bid() external payable {

        //省略其它逻辑
        ...
        highestBidder = msg.sender;
        highestBid = msg.value;
        emit HighestBidIncreased(msg.sender, msg.value);
    }

我们可以通过emit调用事件方法,然后这个事件就作为日志记录到了以太坊区块链中。日志是以太坊区块链中一种特殊的数据结构,你可以把它当作区块链的一部分,只要区块链在,日志就在。日志和产生它的智能合约的地址事绑定的。

写了日志有啥用呢?它的作用就是可以被订阅。很多智能合约项目都是传统的web项目 智能合约的这种架构,业务系统有些在链外,那么链上发生的事情就可以基于这种发布订阅机制进行通知,从而打通链上和链下。

举个例子,

代码语言:javascript复制
contract ClientReceipt {
    event Deposit(
        address indexed from,
        bytes32 indexed id,
        uint value
    );

    function deposit(bytes32 id) public payable {
        emit Deposit(msg.sender, id, msg.value);
    }
}

上面定义了一个存钱的合约,存完钱发布一个事件。

然后我们可以通过javascript(web3.js)订阅这个事件,

代码语言:javascript复制
var abi = /* abi as generated by the compiler */;
var ClientReceipt = web3.eth.contract(abi);
var clientReceipt = ClientReceipt.at("0x1234...ab67" /* address */);

var depositEvent = clientReceipt.Deposit();

// watch for changes
depositEvent.watch(function(error, result){
    // result contains non-indexed arguments and topics
    // given to the `Deposit` call.
    if (!error)
        console.log(result);
});

我们再稍微深入一点到虚拟机层面。我们知道以太坊的虚拟机是EVM,EVM的指令有几种和日志相关的操作码,分别是:LOG0,LOG1 , LOG2 , LOG3 和 LOG4。

日志的数据结构包含主题和数据两个部分,这里的1234代表的事日志包含的主题(topic)数量。如果你用过类似kafka,rocketmq等消息中间价一定对主题这个概念不陌生。主题是用来简单描述事件(日志)的,也可以用来过滤或者搜索日志。

比如我们有这样一个事件定义:

代码语言:javascript复制
event Transfer(address indexed _from, address indexed _to, uint256 _value);

用indexed修饰的参数会被作为主题,而没有修饰的参数会作为日志的数据部分。在这个例子里面,意味着转账的发送方和接收方可以被搜索,而转账金额不能被搜索。

0 人点赞