当`require`评估为false时如何继续执行函数

问题描述

我有一个关于修饰符、要求和函数流的问题。

这是一个简单的例子来说明我的观点。

contract Numbers {
    uint256[] private _numbers;

    modifier bigNumber(uint256 someNumber) {
        require(someNumber > 10,"Numbers: Number must be greater than 10");
        _;
    }

    function _addNumber(uint256 someNumber) private bigNumber(someNumber) {
        _numbers.push(someNumber);
    }

    function addNumbers(uint256[] memory _newNumbers) external {
        for (uint256 i = 0; i < _newNumbers.length; i++) {
            _addNumber(_newNumbers[i]);
        }
    }
}

假设我用 addNumbers([11,12,13,8,20,21,5,22,23]);

调用这个函数

到达 8 时我会收到错误消息。

有没有办法使用以下值继续运行该函数

我之前曾想过添加一个 if 来“捕获”错误,但也许有更好的方法

提前致谢!

解决方法

失败的 require() 条件引发异常。


就您而言,最简单的方法是在您的 addNumbers() 函数中复制条件。

function addNumbers(uint256[] memory _newNumbers) external {
    for (uint256 i = 0; i < _newNumbers.length; i++) {

        // duplicated condition from the modifier
        if (_newNumbers[i] <= 10) {
            _addNumber(_newNumbers[i]);
        }
    }
}

如果 _addNumber() 函数是 external(即可以从合约外部执行),您可以使用 try/catch 表达式。

// mind the changed visibility from `private` to `external`
function _addNumber(uint256 someNumber) external bigNumber(someNumber) {
    _numbers.push(someNumber);
}

function addNumbers(uint256[] memory _newNumbers) external {
    for (uint256 i = 0; i < _newNumbers.length; i++) {
        try this._addNumber(_newNumbers[i]) {
            // do nothing if didn't throw an exception
        } catch (bytes memory) {
            // do nothing if threw an exception
        }
    }
}

Try/catch 目前 (v0.8) 仅支持外部调用和合约创建。这就是为什么您需要将 _addNumber() 函数置于外部以使其工作。

文档:https://docs.soliditylang.org/en/v0.8.6/control-structures.html#try-catch

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...