20. solidity 发送ETH:call()

20. 发送ETH:call()

Solidity有三种方法向其他合约发送ETH,他们是:transfer()send()call(),其中call()是被鼓励的用法。

  • call没有gas限制,最为灵活,是最提倡的方法;
  • transfer2300 gas限制,但是发送失败会自动revert交易,是次优选择;
  • send2300 gas限制,而且发送失败不会自动revert交易,几乎没有人用它。

接收ETH合约

我们先部署一个接收ETH合约ReceiveETHReceiveETH合约里有一个事件Log,记录收到的ETH数量和gas剩余。还有两个函数,一个是receive()函数,收到ETH被触发,并发送Log事件;另一个是查询合约ETH余额的getBalance()函数。

contract ReceiveETH {
    // 收到eth事件,记录amount和gas
    event Log(uint amount, uint gas);
    
    // receive方法,接收eth时被触发
    receive() external payable{
        emit Log(msg.value, gasleft());
    }
    
    // 返回合约ETH余额
    function getBalance() view public returns(uint) {
        return address(this).balance;
    }
}

部署ReceiveETH合约后,运行getBalance()函数,可以看到当前合约的ETH余额为0

20-1

发送ETH合约

我们将实现三种方法向ReceiveETH合约发送ETH。首先,先在发送ETH合约SendETH中实现payable构造函数receive(),让我们能够在部署时和部署后向合约转账。

contract SendETH {
    // 构造函数,payable使得部署的时候可以转eth进去
    constructor() payable{}
    // receive方法,接收eth时被触发
    receive() external payable{}
}

call:接收方地址.call{value: 发送ETH数额}(“”)

  • 用法是接收方地址.call{value: 发送ETH数额}("")
  • call()没有gas限制,可以支持对方合约fallback()receive()函数实现复杂逻辑。
  • call()如果转账失败,不会revert
  • call()的返回值是(bool, data),其中bool代表着转账成功或失败,需要额外代码处理一下。

代码样例:

// call()发送ETH
function callETH(address payable _to, uint256 amount) external payable{
    // 处理下call的返回值,如果失败,revert交易并发送error
    (bool success,) = _to.call{value: amount}("");
    if(!success){
        revert CallFailed();
    }
}

ReceiveETH合约发送ETH,此时amount为10,value为0,amount>value,转账失败,因为经过处理,所以发生revert

20-7

此时amount为10,value为11,amount<=value,转账成功。

20-8

transfer:接收方地址.transfer(发送ETH数额)

  • 用法是接收方地址.transfer(发送ETH数额)
  • transfer()gas限制是2300,足够用于转账,但对方合约的fallback()receive()函数不能实现太复杂的逻辑。
  • transfer()如果转账失败,会自动revert(回滚交易)。

代码样例,注意里面的_toReceiveETH合约的地址,amountETH转账金额:

// 用transfer()发送ETH
function transferETH(address payable _to, uint256 amount) external payable{
    _to.transfer(amount);
}

部署SendETH合约后,对ReceiveETH合约发送ETH,此时amount为10,value为0,amount>value,转账失败,发生revert

20-2

此时amount为10,value为10,amount<=value,转账成功。

20-3

ReceiveETH合约中,运行getBalance()函数,可以看到当前合约的ETH余额为10

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-dLmLE3km-1680011371010)()]

send:接收方地址.send(发送ETH数额)

  • 用法是接收方地址.send(发送ETH数额)
  • send()gas限制是2300,足够用于转账,但对方合约的fallback()receive()函数不能实现太复杂的逻辑。
  • send()如果转账失败,不会revert
  • send()的返回值是bool,代表着转账成功或失败,需要额外代码处理一下。

代码样例:

// send()发送ETH
function sendETH(address payable _to, uint256 amount) external payable{
    // 处理下send的返回值,如果失败,revert交易并发送error
    bool success = _to.send(amount);
    if(!success){
        revert SendFailed();
    }
}

ReceiveETH合约发送ETH,此时amount为10,value为0,amount>value,转账失败,因为经过处理,所以发生revert

20-5

此时amount为10,value为11,amount<=value,转账成功。

20-6

习题

  1. vitalik写了一个合约,并且该合约在被部署时可以转ETH进去,那么该合约的构造函数可以是:

    constructor() payable{}
    
  2. vitalik写了一个用send()发送ETH的函数:
    error SendFailed(); 
    function sendETH(address payable _to, uint256 amount) external payable{
                ______________________________________
        }
    该函数执行失败时会自动revert交易,那么填入横线处的代码可以是:
    bool success = _to.send(amount); if( !success ){ revert SendFailed(); }
    
  3. vitalik又写了一个用call()发送ETH的函数:
    error CallFailed();
    function callETH(address payable _to, uint256 amount) external payable{
                ______________________________________
        }
    该函数执行失败时会自动revert交易,那么填入横线处的代码可以是:
    (bool success,) = _to.call{value: amount}(" "); if( !success ){ revert CallFailed(); }
    
  4. 假设存在如下图所示的两个合约(sendETH和ReceiveETH),两个合约目前ETH余额皆为0,现在vitalik想通过SendETH合约的callETH函数往ReceiveETH合约转入1ETH,他将交易的value设置为2ETH,同时交易成功执行,那么此时sendETH合约和ReceiveETH的ETH余额分别为:1ETH;1ETH

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-nICk9N39-1680011373607)(null)]