Solidity智能合约中的异常处理(error、require 和 assert)

Solidity 中的三种抛出异常方法:errorrequireassert


在 Solidity 开发中,异常处理是确保智能合约安全性和正确性的关键步骤。Solidity 提供了三种主要方法来抛出异常:errorrequireassert。本文将详细介绍这三种方法的用途、实现方式及其各自的特点,并对它们的 Gas 消耗进行比较。


目录

  1. Solidity 中的异常处理
    1.1 什么是异常?
    1.2 异常处理的必要性
    1.3 Solidity 异常的常见场景

  2. error:自定义错误
    2.1 error 的定义
    2.2 error 的使用场景
    2.3 error 的语法
    2.4 error 的Gas消耗

  3. require:前置条件检查
    3.1 require 的作用
    3.2 require 的语法
    3.3 require 的常见使用场景
    3.4 require 的 Gas 消耗

  4. assert:不变量检查
    4.1 assert 的作用
    4.2 assert 的语法
    4.3 assert 的使用场景
    4.4 assert 的 Gas 消耗

  5. 三者的对比与最佳实践
    5.1 功能对比
    5.2 安全性对比
    5.3 Gas 消耗对比

  6. 总结

在这里插入图片描述

1. Solidity 中的异常处理

1.1 什么是异常?

异常是指在程序运行过程中发生的不正常或意外的情况。在 Solidity 中,异常通常指程序遇到错误条件时的中断执行。

1.2 异常处理的必要性

在智能合约中,异常处理的目标是确保交易不会在有错误的情况下继续执行,以防止状态被意外更改或资金被错误转移。

1.3 Solidity 异常的常见场景

  • 用户输入不合法(如溢出、负值等)
  • 外部合约调用失败
  • 合约逻辑中的不变量遭到破坏
  • 资金不足或无法执行转账

2. error:自定义错误

2.1 error 的定义

error 是 Solidity 0.8.4 版本引入的新特性,允许开发者定义自定义错误。自定义错误为错误报告提供了更多的灵活性,并且能够节省 Gas。

2.2 error 的使用场景

自定义错误主要用于需要抛出特定的异常并提供更详细的错误信息的场景。它相比传统的异常处理方式,可以节省 Gas,尤其是在复杂合约中。

2.3 error 的语法

// 定义错误
error InsufficientBalance(uint requested, uint available);contract Token {function withdraw(uint amount) public {if (amount > address(this).balance)revert InsufficientBalance({requested: amount,available: address(this).balance});// 继续执行其他逻辑}
}

2.4 error 的Gas消耗

使用 error 定义自定义错误时,相比 requireassert,通常会节省更多的 Gas,尤其是当错误需要包含复杂数据时。由于错误消息不作为字符串存储,它的处理更加高效。


3. require:前置条件检查

3.1 require 的作用

require 用于在合约执行之前检查某些条件是否成立,通常用于验证输入参数或外部合约调用结果。

3.2 require 的语法

function transfer(address recipient, uint amount) public {require(amount <= balance, "Insufficient balance");// 执行转账
}

3.3 require 的常见使用场景

  • 检查调用方是否具有足够的权限
  • 验证输入数据的合法性
  • 验证外部合约的返回值

3.4 require 的 Gas 消耗

require 语句会消耗一定的 Gas,但由于 require 在条件不满足时立即中断执行,未使用的 Gas 会被退还。因此,require 适合用于条件检查时。


4. assert:不变量检查

4.1 assert 的作用

assert 用于检查代码逻辑中的不变量,即程序在任何时候都应该满足的条件。如果 assert 失败,意味着代码中存在致命的错误。

4.2 assert 的语法

uint x = 0;function increment() public {x += 1;assert(x > 0); // 确保 x 永远大于 0
}

4.3 assert 的使用场景

  • 用于捕捉代码中的严重错误,特别是不应该发生的逻辑错误。
  • 检查合约中的状态是否在预期范围内。

4.4 assert 的 Gas 消耗

assert 失败时会消耗所有剩余的 Gas,因为它通常用来捕捉不可预见的严重错误。因此,应慎用 assert,只在关键性逻辑的检查中使用。


5. 三者的对比与最佳实践

5.1 功能对比

  • error 提供了更灵活的错误报告机制,适合复杂错误处理。
  • require 适用于输入验证和外部合约结果检查。
  • assert 主要用于捕获不可预见的内部错误或逻辑漏洞。

5.2 安全性对比

  • errorrequire 通常用于用户或合约交互时的错误检查。
  • assert 应用于确保内部逻辑的不变量,更多用于调试目的。

5.3 Gas 消耗对比

  • error:节省 Gas,尤其是复杂的错误处理。
  • require:较为高效,未使用的 Gas 可退还。
  • assert:在失败时消耗所有 Gas,应用场景更局限。

6. 总结

Solidity 提供了 errorrequireassert 三种异常处理方式,每种方式都有其特定的应用场景。开发者应根据合约的实际需求和安全性要求,选择适合的异常处理机制。此外,Gas 消耗的比较也提示我们在大多数情况下,应优先使用 require 进行输入检查,使用 error 进行复杂错误处理,而 assert 应仅用于关键性的不变量检查。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/434993.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

nginx+keepalived健康检查案例详解(解决nginx出现故障却不能快速切换到备份服务器的问题)

文章目录 简介配置过程前置环境请看创建健康检查脚本结果测试 简介 在我们通过nginxkeepalived实现高可用后&#xff0c;会发现nginx出现故障的时候keepalived并不会将虚拟ip切换到备份服务器上其原理就是nginx和keepalived是两个独立的服务&#xff0c;Nginx的故障状态不会触…

使用Docker-Compose部署SpringBoot项目的案例

Docker-Compose是Docker官方的一个开源项目&#xff0c;主要用于实现对Docker容器集群的快速编排和管理。该项目由Python编写&#xff0c;通过调用Docker服务提供的API来管理容器。只要所操作的平台支持Docker API&#xff0c;就可以利用Docker-Compose进行编排管理。Docker-Co…

【微服务即时通讯系统】——brpc远程过程调用、百度开源的RPC框架、brpc的介绍、brpc的安装、brpc使用和功能测试

文章目录 brpc1. brpc的介绍1.1 rpc的介绍1.2 rpc的原理1.3 grpc和brpc 2. brpc的安装3. brpc使用3.1 brpc接口介绍 4. brpc使用测试4.1 brpc同步和异步调用 brpc 1. brpc的介绍 1.1 rpc的介绍 RPC&#xff08;Remote Procedure Call&#xff09;远程过程调用&#xff0c;是一…

X86下一文带你构建Apollo9.0运行环境(基于Ubuntu20.04避坑版)

X86下一文带你构建Apollo9.0运行环境基于Ubuntu20.04避坑版 前言准备安装基础软件1.安装Docker19.03安装Nvidia驱动安装配置Nvidia container toolkit 下载Apollo源码&#xff08;笔者下载的是releases下9.0.0版本&#xff0c;大家可以参考&#xff09;编译Apollo9.0下载资源包…

如何使用GitHub Desktop管理GitLab库

不管是新手还是老手&#xff0c;Github Desktop都是在苹果系统和Windows系统上管理与创建项目的不错的方式&#xff0c;GitHub Desktop都能够让在GitHub上的工作流更为简单快捷。 注意&#xff0c;以下步骤只支持原版的GitHub Desktop 第一步 从这下载GitHub Desktop打开你的G…

【hot100-java】【柱状图中最大的矩形】

R9-栈篇 面积最大矩形的高度一定是 heights 中的元素 简单解释&#xff0c;就是说&#xff0c;最大高度必然是heights中的一个元素&#xff0c;我们假设是h&#xff0c;然后我们基于h&#xff0c;左右拓展&#xff0c;尽量拓展到h越来越高&#xff08;符合单调栈&#xff09;&a…

手机改IP地址怎么弄?全面解析与操作指南

在当今数字化时代&#xff0c;IP地址作为设备在网络中的唯一标识&#xff0c;其重要性不言而喻。有时候&#xff0c;出于隐私保护、网络访问需求或其他特定原因&#xff0c;我们可能需要更改手机的IP地址。然而&#xff0c;对于大多数普通用户来说&#xff0c;如何操作可能还是…

Redis 篇-深入了解 Redis 中的 RESP 通信协议与内存回收(过期 key 处理、内存淘汰策略)

&#x1f525;博客主页&#xff1a; 【小扳_-CSDN博客】 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 文章目录 1.0 Redis 通信协议 - RESP 协议 2.0 Redis 内存回收 2.1 Redis 内存回收 - 过期 key 处理 2.1.1 Redis 是如何知道一个 Key 是否过期呢&#xff1f; 2.1.2 是不是 TT…

Linux基础(三):安装CentOS7(系统安装+桥接联网+换源)

1.分区设置 由于使用 GPT 的关系&#xff0c; 因此根本无须考虑主/延伸/逻辑分区的差异。CentOS 默认使用 LVM 的方式来管理你的文件系统。使用GPT进行分区&#xff1a; 开机管理程序&#xff08; boot loader&#xff09; 使用CentOS 7.x默认的grub2软件。 2.各种分区格式 …

DataLight(V1.4.5) 版本更新,新增 Ranger、Solr

DataLight&#xff08;V1.4.5&#xff09; 版本更新&#xff0c;新增 Ranger、Solr DataLight 迎来了重大的版本更新&#xff0c;现已发布 V1.4.5 版本。本次更新对平台进行了较多的功能拓展和优化&#xff0c;新增了对 Ranger 和 Solr 服务组件的支持&#xff0c;同时对多项已…

Oracle 配置恢复目录catalog

一.介绍 Oracle中使用RMAN备份的数据我们分为两类 RMAN知识库数据库的数据块 Oracle默认把 RMAN知识库 放在目标数据库的控制文件中&#xff0c;在以后进行恢复的时候 我们要先读知识库的信息然后才能恢复。 但这样就产生了一个问题&#xff0c;知识库放在了控制文件上&#xf…

【C++】set与map

目录 一、预备知识&#xff1a; 1、关联式容器&#xff1a; 2、键值对&#xff1a; 3、树形结构的关联式容器&#xff1a; 二、set&#xff1a; 1、set的介绍&#xff1a; 2、使用&#xff1a; 1、set的构造&#xff1a; 2、set的各种功能&#xff1a; 3、multiset 三…

AOP-代理实现

三种代理实现 1 JDK动态代理实现-基于接口代理 2 CGLIB动态代理实现-基于类代理 3 AspectJ 适配实现 为什么Proxy.newProxyInstance 会生成新的字节码&#xff1f; 创建代理类&#xff1a; Proxy.newProxyInstance 首先会检查缓存中是否有已存在的代理类字节码。 如果没有&…

计算机毕业设计 C语言学习辅导网站的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

14.安卓逆向-frida基础-编写hook脚本2

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a;图灵Python学院 本人写的内容纯属胡编乱造&#xff0c;全都是合成造假&#xff0c;仅仅只是为了娱乐&#xff0c;请不要盲目相信。 工…

16. C++ TinyWebServer项目总结(16. 服务器调制、调试和测试)

主要包括&#xff1a; 使用 tcpdump 抓包&#xff1b;使用 gdb 调试器&#xff1b;使用压力测试工具&#xff0c;模拟现实世界中的高并发请求&#xff0c;测试服务器在高压状态下的稳定性。 最大文件描述符数 Linux 对应用进程能打开的最大文件描述符数量有两个层次的限制&a…

node的版本管理工具volta

安装方式 # mac curl https://get.volta.sh | bash # Windows Installation winget install Volta.Volta切换版本 volta install node指定版本根据项目固定node和包管理器版本和 该命令会在package.json生成volta的配置&#xff0c;volta会自动读取项目的该配置来决定node的…

【STM32】TCP/IP通信协议--LWIP内存管理

五、LWIP内存管理 1.什么是内存管理&#xff1f; &#xff08;1&#xff09;内存管理&#xff0c;是指软件运行时对计算机内存资源的分配的使用的技术&#xff0c;其主要目的是如何高效、快速的分配&#xff0c;并且在适当的时候释放和回收内存资源&#xff08;就比如C语言当…

安全的价值:构建现代企业的基础

物理安全对于组织来说并不是事后才考虑的问题&#xff1a;它是关键的基础设施。零售商、医疗保健提供商、市政当局、学校和所有其他类型的组织都依赖安全系统来保障其人员和场所的安全。 随着安全技术能力的不断发展&#xff0c;许多组织正在以更广泛的视角看待他们的投资&am…

SQL学习1

24.9.28学习目录 一.数据库1.SQL语句基础2.匹配条件 一.数据库 对于嵌入式的数据库&#xff0c;其使用的是SQLite这种小型数据库&#xff1b; 在ubuntu中的下载方法 //字符界面 sudo apt-get install sqlite3//图形界面 sudo apt-get install sqlitemanSQLite特点&#xff1a…