关于函数名直接调用与call调用的一个实验

本文探讨了Solidity中call函数调用合约失败时的回滚机制,通过对比call函数与直接调用函数的区别,解释了为何前者不会导致主合约回滚,并通过实例验证了这一现象。关键发现是:call函数不创建新的调用帧,因此其失败会回滚,而直接调用则会遵循不同的规则。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在研究solidity的过程中有个疑问,都说使用call函数调用合约函数,如果调用失败只会返回false,并不会回滚。这里的失败是指什么?是指被调用合约函数回滚吗?那么主调用合约会跟着回滚吗?如果使用函数名直接调用又会怎样?来做个实验看看。

假设有First、Second两个合约,如下:

// SPDX-License-Identifier: GPL-3.0
pragma solidity ^0.5.0;

contract First {

    uint public data = 1;
    //Second calledContract;
    address _addr;

    constructor (address addr) public {
        //calledContract = Second(addr);
        _addr = addr;
    }

    function testFunctionCall() public returns(bool){

        data = 2;
        //calledContract.revertFunc();
        bytes4  SELECTOR = bytes4(keccak256(bytes("revertFunc()")));
        bytes memory argument = abi.encodeWithSelector(SELECTOR);
        (bool success, ) = _addr.call(argument);
        return success;

    }

}

contract Second {

    function revertFunc() external {
        revert();
    }

}

把Second合约的地址作为参数传递给First的构造函数,然后在First合约中通过call函数调用Second合约中的回滚测试函数revertFunc(),结果显示,First合约中的状态变量data的值变为了2,说明First合约没有回滚。

下面使用函数名直接调用测试一下:

// SPDX-License-Identifier: GPL-3.0
pragma solidity ^0.5.0;

contract First {

    uint public data = 1;
    Second calledContract;
    //address _addr;

    constructor (address addr) public {
        calledContract = Second(addr);
        //_addr = addr;
    }

    function testFunctionCall() public {

        data = 2;
        calledContract.revertFunc();
        //bytes4  SELECTOR = bytes4(keccak256(bytes("revertFunc()")));
        //bytes memory argument = abi.encodeWithSelector(SELECTOR);
        //(bool success, ) = _addr.call(argument);
        //return success;

    }

}

contract Second {

    function revertFunc() external {
        revert();
    }

}

结果显示First合约中的状态变量data的值仍然为1,同时testFunctionCall也执行失败了,这说明First合约状态回滚了。

结论:

使用call函数调用合约函数与直接使用函数名调用在底层实现上应该是不一样的,有的文档中说call函数会重新建立一个调用帧,那么我猜测使用函数名直接调用这样的方式并不会创建一个新的调用帧,所以调用函数也会失败并回滚。

这个实验也解决了我的一个担忧,我总担心某个通过函数名的直接调用失败后,调用合约的状态变量是否会回滚,现在我知道了,应该是回滚的。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值