发送一笔以太坊网络转账交易,除了成功之外,会遇到各种问题,比如发送的一笔交易在etherscan中查不到,或者查到是pending的状态,后来过了几个小时消失了,这些都与节点的交易池的处理逻辑相关,而交易池的逻辑可以有各种实现,目前官方的go-ethereum 的交易池逻辑如下。
当一条交易进入挖矿节点的交易池(tx_pool)时,此节点会做出以下逻辑:
通过判断交易hash,当这条交易已经在节点的交易池里面时,就会丢弃掉当前收到的这一个交易。
如果是一个全新的hash,就会更具共识协议,对这一条交易做基本的验证。验证包括:长度,value,是否溢出当前区块的gaslimit,Nonce值,转账提供的gas大小是否太小等。
如果验证不通过就会返回相应的错误代码。当验证通过的话,需要根据当前交易池的状态来决定,如果交易池满了的话,那么判断当前的转账交易提供的gas是否高于交易池里当前提供gas最低的那条交易。如果低了,就返回错误
ErrUnderpriced
,这是返回的错误经常在etherscan中看到。如果当前交易提供的gas值高的话,那么就剔除掉了那一条提供最低gas的转账交易,为当前交易腾出空间。这种情况发生之后,往往在etherscan中会观察到正在pending的交易消失了,找不到了。(也有很大机率依然能够查到pending的交易,原因是etherscan连接了很多节点,每个节点的交易池的状态都是不一样的,那条被踢出的交易在别的节点依然pending)如果当前交易的Nonce已经在交易池里面了,说明了这个用户想替代之前发出的相同的Nonce的交易。这时节点会判断当前交易的gas是否高出上一条相同Nonce的交易gas某一个阈值(比如默认的是10%)。如果高出了,那么就会剔除之前的那个交易,新的交易就会保留,如果没有高出,当前的交易就会返回失败。
以上逻辑都通过了,那么此条交易便成功存在交易池中等待打包,当然也有可能会被别的高gas的转账交易剔除的概率。
总结,如果你发送了一笔交易碰巧赶上了以太坊网络拥堵,那么你的交易很有可能发送失败。原因是所有的挖矿节点都不接受这么低的gas交易了,或者你本来在交易池中,被剔除了。