如何使用 geth1.8来监听合约事件
新功能介绍
geth1.8版本带来了新的事件处理方式,使用 abigen 可以自动生成包含合约事件监听以及过滤相关代码.
这样就不用自己去写代码解析 log. 比如:abigen --sol token.sol --pkg token --out token.go
一个例子
通过监听谁给我转账,来说明如何使用新的接口
创建 Filter
只需指定合约地址即可.
filter, err := token.NewTokenFilterer(tokenAddr, c)
监听将要发生的事件
这个应该放在过滤历史事件之前,因为有可能在处理历史事件过程中产生了新的事件.如果顺序错了,就会造成事件丢失.
ch := make(chan *token.TokenTransfer, 10) sub, err := filter.WatchTransfer(nil, ch, nil, []common.Address{toAddr}) if err != nil { log.Fatalf("watch transfer err %s", err) } go func() { for { select { case <-sub.Err(): return case e := <-ch: log.Printf("new transfer event from %s to %s value=%s,at %d", e.From.String(), e.To.String(), e.Value, e.Raw.BlockNumber) } } }()
简单直观,不用去关心 log 的细节.
感兴趣的话,可以看一下 TokenTranser 结构// TokenTransfer represents a Transfer event raised by the Token contract.type TokenTransfer struct { From common.Address To common.Address Value *big.Int Raw types.Log // Blockchain specific contextual infos
过滤历史事件
也很直观,把你感兴趣的事件范围传递进去,会返回一个 Iterator, 遍历就 ok 了.
history, err := filter.FilterTransfer(&bind.FilterOpts{Start: 480000}, nil, []common.Address{toAddr}) for history.Next() { e := history.Event log.Printf("%s transfer to %s value=%s, at %d", e.From.String(), e.To.String(), e.Value, e.Raw.BlockNumber) }
结论
有了这些自动生成的代码以后,我们就不用费劲去理解过滤时候的 Topic 怎么设置,Log怎么解析. 直接关注我们想要的事件本身就可以了.
当然也不是没有问题,如果我关注的不是某个合约上发生了转账事件,而是所有的ERC20token, 那么该怎么写呢? 目前我是没想到怎么实现,要想这么做还是要回到老办法上.