智能合约语言 Solidity - 错误处理
Posted liankexing
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了智能合约语言 Solidity - 错误处理相关的知识,希望对你有一定的参考价值。
大家好,这里是链客区块链技术问答社区,链客 ,有问必答!!
Solidity 是以太坊智能合约编程语言,阅读本文前,你应该对以太坊、智能合约有所了解,如果你还不了解,建议你先看以太坊是什么?
什么是错误处理
错误处理是指在程序发生错误时的处理方式,Solidity处理的方式和常见的语言不一样,其时通过回退状态的方式来处理错误,发生异常时会撤销当前调用所改变的状态,同时给调用者返还一个错误标识。
为什么要这么设计呢?
我们把区块链理解为全球分享的分布式事务性数据库,全球共享意味着每个人都可以读取其中记录,想要修改这个数据库的内容,就必须创建一个事务,意味着要的修改只能被完全应用或不进行。
如何处理
Solidity提供2个函数assert和requrie进行检查,条件如果不满足则抛出异常,assert函数通常用来检查内部错误,另一个用来检查输入变量或合同状态变量是否满足条件及验证调用外部合约返回值。如果正确使用assert,由一个solitity分析工具就可以分析出只能合约中的错误,帮助我们发现合约中逻辑错误的bug。
另外还有两种方式来触发异常:
revert函数可以用来标记错误并回退当前调用
使用throw关键字抛出异常
当子调用中发生异常时,异常会自动向上“冒泡”。 不过也有一些例外:send,和底层的函数调用call, delegatecall,callcode,当发生异常时,这些函数返回false。
在下面通过一个示例来说明如何使用require来检查输入条件,以及assert用于内部错误检查:
pragma solidity ^0.4.0;
contract Sharer {
function sendHalf(address addr) public payable returns (uint balance) {
require(msg.value % 2 == 0); // 仅允许偶数
uint balanceBeforeTransfer = this.balance;
addr.transfer(msg.value / 2); // 如果失败,会抛出异常,下面的代码就不是执行
assert(this.balance == balanceBeforeTransfer - msg.value / 2);
return this.balance;
}
}
我们实际运行下,看看异常是如何发生的:
首先打开Remix,贴入代码,点击创建合约。如下图:
运行测试1:附加1wei (奇数)去调用sendHalf,这时会发生异常,如下图:
运行测试2:附加2wei 去调用sendHalf,运行正常。
运行测试3:附加2wei以及sendHalf参数为当前合约本身,在转账是发生异常,因为合约无法接收转账,错误提示上图类似。
assert类型异常
在下述场景中自动产生assert类型的异常:
如果越界,或负的序号值访问数组,如i >= x.length 或 i < 0时访问x[i]
如果序号越界,或负的序号值时访问一个定长的bytesN。
被除数为0, 如5/0 或 23 % 0。
对一个二进制移动一个负的值。如:5<<i; i为-1时。
require类型异常
在下述场景中自动产生require类型的异常:
调用throw
如果调用require的参数为false
如果在使用new创建一个新合约时出现第3条的原因没有正常完成。
如果调用外部函数调用时,被调用的对象不包含代码。
当发生require类型的异常时,Solidity会执行一个回退操作(指令0xfd)。
当发生assert类型的异常时,Solidity会执行一个无效操作(指令0xfe)。
在上述的两种情况下,EVM都会撤回所有的状态改变。是因为期望的结果没有发生,就没法继续安全执行。注意assert类型的异常会消耗掉所有的gas, 而require从大都会版本(Metropolis, 即目前主网所在的版本)起不会消耗gas。
以上是关于智能合约语言 Solidity - 错误处理的主要内容,如果未能解决你的问题,请参考以下文章
智能合约Solidity语言错误处理函数(requirerevertassert)使用详解
智能合约语言 Solidity 教程系列8 - Solidity API(特殊的变量及函数)