问题描述
这是一个简单的例子来说明我的观点。
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