当前位置: 首页 > article >正文

Smart contract -- 自毁合约

        在区块链开发中,Solidity 语言提供了强大的功能,其中自毁合约是一个独特且重要的特性。今天,就让我们深入探讨一下 Solidity 中的自毁合约,以及如何使用 selfdestruct 函数。

         注意:使用继承时请确保代码的正确性,以防丢失个人财产,在这里友情提示您,不要复制来源不明的solidity代码并进行部署。本文为自己梳理总结,如有不足还请指出,感谢包容。 

        学习更多solidity知识请访问 Github -- solidity基础 ,更多实例在 Smart contract

一、自毁合约的概念

         一种具有自我终结能力的智能合约。自毁合约,顾名思义,是指合约在执行过程中,可以主动销毁自身。一旦合约自毁,其占用的存储空间将被释放,同时可以将剩余的以太币发送到指定地址。这一功能在某些特定场景下非常有用,比如当合约完成其使命,或者需要紧急停止合约运行并回收资源时。

       当满足特定条件时,合约可以调用selfdestruct函数,这将导致合约从以太坊区块链上永久删除。合约的存储数据将被清除,并且其关联的代码也不再存在。同时,合约剩余的以太币余额会被发送到指定的接收地址。 

二、自毁合约的用途

  1. 资金清算:当一个项目结束或合约不再需要时,可以使用自毁合约将剩余资金返还给所有者或特定的受益人。例如,一个众筹合约在达到目标金额并完成项目交付后,可能会选择自毁并将剩余资金退还给参与者。

  2. 安全考量:在某些情况下,如果发现合约存在严重漏洞或安全隐患,自毁合约可以作为一种紧急措施,防止黑客进一步攻击和窃取资金。通过自毁合约,可以迅速停止合约的运行,并将资金转移到安全地址。

  3. 合约升级与替换:在智能合约的开发过程中,可能需要对合约进行升级以添加新功能或修复漏洞。旧版本的合约可以通过自毁的方式,将资金和相关状态转移到新版本的合约中,实现平滑过渡。

三、selfdestruct 函数的用法

selfdestruct 是 Solidity 提供的一个内置函数,用于销毁当前合约。它的语法如下:

selfdestruct(address payable recipient)

        其中,recipient 是接收合约剩余以太币的地址,且该地址必须是 payable 类型,这意味着它能够接收以太币。

四、示例代码解析

下面是一个简单的自毁合约示例:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.8;contract SelfDestructContract {constructor() payable {}function destroySelf() external {selfdestruct(payable(msg.sender));}function testCall() external pure returns (uint) {return 123;}
}

合约说明

  1. 构造函数constructor() payable 表示合约部署时可以接收以太币。

  2. destroySelf 函数:这是自毁函数,当被外部调用时,会销毁合约,并将合约中的剩余以太币发送给调用者(msg.sender)。注意,msg.sender 需要被转换为 payable 类型。

  3. testCall 函数:一个简单的测试函数,用于演示合约在自毁前后的状态变化。

五、使用场景与注意事项

使用场景

  • 紧急停止:当合约出现安全漏洞或紧急情况时,可以通过自毁合约来停止其运行,防止进一步的损失。

  • 资源回收:当合约完成其任务后,自毁可以释放区块链上的存储空间,并将剩余资金返还给指定地址。

注意事项

  • 不可逆性:合约一旦自毁,将无法恢复。因此,在调用 selfdestruct 之前,必须确保这是合约的最终状态。

  • 资金处理:自毁时,合约中的剩余资金会发送到指定地址。需要确保该地址能够正确接收和处理这些资金。

  • 状态变化:自毁后,合约中的所有状态变量和存储数据都将被清除。

六、实际应用示例

        假设我们有一个限时任务合约,任务完成后合约自动销毁,并将剩余资金返还给创建者。以下是实现这一功能的代码:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.8;contract TaskContract {address payable public owner;uint public deadline;constructor(uint _duration) payable {owner = payable(msg.sender);deadline = block.timestamp + _duration;}function performTask() external {// 执行任务的逻辑}function withdraw() external {require(msg.sender == owner, "Only owner can withdraw");payable(owner).transfer(address(this).balance);}function destroyContract() external {require(block.timestamp >= deadline || msg.sender == owner, "Cannot destroy yet");selfdestruct(owner);}
}

合约说明

  1. 构造函数:部署合约时,指定任务的持续时间 _duration,并将创建者设置为所有者 owner

  2. performTask 函数:用于执行任务的逻辑。

  3. withdraw 函数:允许所有者提取合约中的资金。

  4. destroyContract 函数:在任务超时或所有者主动调用时,销毁合约,并将剩余资金发送给所有者。

以下是一个完整的自毁合约示例:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.8;// 修改合约名称为 SelfDestructContract
contract SelfDestructContract {constructor() payable {}// 修改函数名称为 destroySelffunction destroySelf() external {selfdestruct(payable(msg.sender));//必须是payable类型,默认是没有的}//一旦调用就会把剩下的主币发送到sender的账户上//自毁的意思也就是说强制把剩下的主币发送到需要的账户上,如果没有会进行报错function testCall() external pure returns (uint) {return 123;}
}

        在这个合约中,SelfDestructContract包含一个构造函数,允许在部署合约时向合约转入资金。destroySelf函数是实现自毁功能的关键,当外部调用destroySelf时,合约将调用selfdestruct函数,并将合约的剩余资金发送给调用者(msg.sender)。testCall函数是一个简单的示例函数,用于返回一个固定的整数值,与自毁功能并无直接关联。

        假设 Alice 部署了这个合约,并向合约转入了 10 以太币。在某个时刻,Alice 决定销毁合约并收回资金,她只需调用合约的destroySelf函数。此时,合约将从区块链上删除,而 10 以太币将被发送回 Alice 的账户。

七、注意事项

  1. 不可逆操作:自毁合约是不可逆的,一旦执行,合约将永久消失,无法恢复。因此,在调用selfdestruct之前,务必仔细确认所有条件和后果。

  2. 安全风险:由于自毁合约涉及资金转移,需要确保接收资金的地址是安全可靠的。如果将资金发送到一个错误或被黑客控制的地址,资金将面临损失风险。

  3. Gas 费用:执行selfdestruct操作需要消耗一定的 Gas 费用。在设计合约时,需要考虑合约的余额是否足够支付 Gas 费用,以确保自毁操作能够成功执行。

八、总结

        selfdestruct 函数是 Solidity 提供的一个强大工具,用于销毁合约并回收资源。在使用时,需要谨慎考虑合约的状态和资金处理,确保自毁操作符合预期且不会导致问题。通过合理的设计和应用场景,自毁合约可以在区块链开发中发挥重要作用,为智能合约的生命周期管理提供灵活性和安全性。

相关文章:

Smart contract -- 自毁合约

在区块链开发中,Solidity 语言提供了强大的功能,其中自毁合约是一个独特且重要的特性。今天,就让我们深入探讨一下 Solidity 中的自毁合约,以及如何使用 selfdestruct 函数。 注意:使用继承时请确保代码的正确性&#…...

unity学习64,第3个小游戏:一个2D跑酷游戏

目录 学习参考 素材资源导入 1 创建项目 1.1 创建1个2D项目 1.2 导入素材 2 背景图bg 2.0 bg素材 2.1 创建背景 2.2 修改素材,且修改摄像机等 2.2.1 修改导入的原始prefab素材 2.2.2 对应调整摄像机 2.2.3 弄好背景 2.3 背景相关脚本实现 2.3.1 错误…...

Python Flask 在网页应用程序中处理错误和异常

Python Flask 在网页应用程序中处理错误和异常 Python Flask 在网页应用程序中处理错误和异常 Python Flask 在网页应用程序中处理错误和异常 在我们所有的代码示例中,我们没有注意如何处理用户在浏览器中输入错误的URL或向我们的应用程序发送错误的参数集的情况。…...

模板方法模式的C++实现示例

核心思想 模板方法设计模式是一种行为设计模式,它定义了一个算法的框架,并将某些步骤的具体实现延迟到子类中。通过这种方式,模板方法模式允许子类在不改变算法结构的情况下重新定义算法的某些步骤。 模板方法模式的核心在于: ​…...

水下机器人推进器PID参数整定与MATLAB仿真

水下机器人推进器PID参数整定与MATLAB仿真 1. PID控制原理 目标:通过调节比例(P)、积分(I)、微分(D)参数,使推进器输出力快速稳定跟踪期望值。传递函数(示例):推进器动力学模型可简化为: [ G(s) = \frac{K}{\tau s + 1} \cdot e^{-Ts} ] 其中:K为增益,τ为时间常…...

在本地部署DeepSeek等大模型时,需警惕的潜在安全风险

在本地部署DeepSeek等大模型时,尽管数据存储在本地环境(而非云端),但仍需警惕以下潜在安全风险: 1. 模型与数据存储风险 未加密的存储介质:若训练数据、模型权重或日志以明文形式存储,可能被物…...

智能焊机监测系统:打造工业安全的数字化盾牌

在现代工业生产中,焊机作为核心设备之一,其稳定性和安全性直接关系到生产效率和产品质量。德州迪格特科技有限公司推出的智能焊机监测系统,通过先进的技术手段,为工业生产构筑了一道坚固的安全防线。 智能监测,保障焊…...

【redis】string类型相关操作:SET、GET、MSET、MGET、SETNX、SETEX、PSETEX

文章目录 二进制存储编码转换SET 和 GETSETGET MSET 和 MGETSETNX、SETEX 和 PSETEX Redis 所有的 key 都是字符串,value 的类型是存在差异的 二进制存储 Redis 中的字符串,直接就是按照二进制数据的方式存储的 不仅仅可以存储文本数据,还可…...

GaussDB安全配置指南:从认证到防御的全方面防护

一、引言 随着企业数据规模的扩大和云端化进程加速,数据库安全性成为运维的核心挑战之一。GaussDB作为一款高性能分布式数据库,提供了丰富的安全功能。本文将从 ​认证机制、权限控制、数据加密、审计日志​ 等维度,系统性地讲解如何加固 Ga…...

总结学习课程

1. 数据加载与预处理 PyTorch工具加载和预处理数据(如MNIST数据集)。 2. 定义模型 - 使用nn.Module构建神经网络,定义各层和前向传播。 3. 损失函数与优化器: 选择损失函数(如交叉熵损失)和优化…...

Ubuntu20.04搭建gerrit code review

一、环境准备 1. 安装 Java 环境‌ Gerrit 依赖 Java 运行环境(推荐 JDK 8): sudo apt install openjdk-11-jdk 验证安装: java -version ‌2. 安装 Git sudo apt install git ‌3. 可选依赖 数据库‌:Gerrit …...

MacOS安装FFmpeg和FFprobe

按照网上很多教程安装,结果都失败了,后来才发现是路径问题,其实安装过程很简单(无奈) 第一步: 在官网下载 打开页面后,可以看到FFmpeg、FFprobe、FFplay和FFserver的下载图标 第二步&#xff1…...

Redis7系列:设置开机自启

前面的文章讲了Redis和Redis Stack的安装,随着服务器的重启,导致Redis 客户端无法连接。原来的是Redis没有配置开机自启。此文记录一下如何配置开机自启。 1、修改配置文件 前面的Redis和Redis Stack的安装的文章中已经讲了redis.config的配置&#xf…...

SpringAI介绍及本地模型使用方法

博客原文地址 前言 Spring在Java语言中一直稳居高位,与AI的洪流碰撞后也产生了一些有趣的”化学反应“,当然你要非要说碰撞属于物理反应也可以, 在经历了一系列复杂的反应方程后,Spring家族的新成员——SpringAI,就…...

Zookeeper实践指南

Zookeeper实践指南 1. 什么是 Zookeeper? Zookeeper 是 Apache 旗下的一个开源分布式协调框架,主要用于解决分布式系统中的一致性问题,提供高效可靠的分布式数据管理能力。 1.1 Zookeeper 的核心特性 顺序一致性:客户端的更新…...

Unity 基础知识总结(持续更新中...)

引擎基础 Unity有哪几个主要窗口? Scene窗口 用于场景搭建和UI界面拼接 Game窗口 游戏运行预览 Hierarchy窗口 查看和调整场景对象层级结构 Project窗口 游戏工程资源 Inspector创建 属性查看器,属性设置、脚本组件挂载 Unity提供了几种光源…...

IDEA接入阿里云百炼中免费的通义千问[2025版]

安装deepseek 上一篇文章IDEA安装deepseek最新教程2025中说明了怎么用idea安装codeGPT插件,并接入DeepSeek,无奈接入的官方api已经不能使用了,所以我们尝试从其他地方接入 阿里云百炼https://bailian.console.aliyun.com/ 阿里云百炼‌是阿…...

中级网络工程师面试题参考示例(1)

一、基础理论 1. OSI七层模型与TCP/IP四层模型的区别是什么?请举例说明第三层(网络层)和第四层(传输层)的核心协议。 参考答案: OSI七层模型分为物理层、数据链路层、网络层、传输层、会话层、表示层、应用…...

主流大语言模型中Token的生成过程本质是串行的

主流大语言模型中Token的生成过程本质是串行的 flyfish 1. 串行生成 自回归模型的核心逻辑: 大模型(如GPT-2)采用自回归架构,每个Token的生成必须基于已生成的完整历史序列。例如,生成“今天天气很好”时&#xff1a…...

3.03-3.09 Web3 游戏周报:Sunflower Land 周留存率 74.2%,谁是本周最稳链游?

回顾上周的区块链游戏概况,查看 Footprint Analytics 与 ABGA 最新发布的数据报告。 【3.03–3.09】Web3 游戏行业动态 Sui 背后开发公司 Mysten Labs 宣布收购游戏开发平台 ParasolYescoin 创始人因合伙人纠纷被警方带走,案件升级为刑事案件Animoca B…...

高级java每日一道面试题-2025年2月18日-数据库篇-MySQL 如何做到高可用方案?

如果有遗漏,评论区告诉我进行补充 面试官: MySQL 如何做到高可用方案? 我回答: 在Java高级面试中,讨论MySQL如何实现高可用性方案是一个重要话题。这不仅涉及到数据库的稳定性和可靠性,还关系到系统的整体性能和用户体验。以下是结合提供的信息进行综…...

【编程题】7-5 堆中的路径

7-5 堆中的路径 1 题目原文2 思路解析3 代码实现 1 题目原文 题目链接:7-5 堆中的路径 将一系列给定数字插入一个初始为空的最小堆 h h h。随后对任意给定的下标 i i i,打印从第 i i i 个结点到根结点的路径。 输入格式: 每组测试第 1 1 1 行包含 …...

Scala 中的访问修饰符

在Scala中,面向对象的权限控制主要通过访问修饰符来实现。Scala提供了以下几种访问修饰符来控制类、对象、成员变量和方法的访问权限: 1. 默认访问权限(无修饰符) 如果没有指定任何访问修饰符,成员默认是public的&…...

flask_restx 定义任意类型参数

之前定义的content只是string,现在需要支持即可以string也可以list from flask_restx import fieldsclass Messages:def get_model(api):return api.model("Message",{"role": fields.String(requiredTrue, description"The role of messa…...

Unity3D网格简化与LOD技术详解

前言 在Unity3D游戏开发中,网格简化(Mesh Simplification)和细节层次(Level of Detail, LOD)技术是优化渲染性能的关键手段,尤其在处理复杂场景和高精度模型时至关重要。这两种技术通过减少模型的几何复杂…...

爬取数据时如何处理可能出现的异常?

在爬取数据时,处理可能出现的异常是确保爬虫稳定运行的关键。以下是一些常见的异常处理策略和具体实现方法,这些方法可以帮助你在爬虫开发中更有效地应对各种问题。 1. 使用 try-catch 块捕获异常 在PHP中,try-catch 块是处理异常的基本工具…...

TCP/IP原理详细解析

前言 TCP/IP是一种面向连接,可靠的传输,传输数据大小无限制的。通常情况下,系统与系统之间的http连接需要三次握手和四次挥手,这个执行过程会产生等待时间。这方面在日常开发时需要注意一下。 TCP/IP 是互联网的核心协议族&…...

MPPT与PWM充电原理及区别详解

MPPT(最大功率点跟踪)和PWM(脉宽调制)是太阳能充电控制器中常用的两种技术,它们在原理、效率和适用场景上有显著区别。以下是两者的详细对比: 1. 工作原理 PWM(脉宽调制) 核心机制…...

数据量过大的时候导出数据很慢

原因解析 速度慢无非两个原因: sql取数很慢程序很慢 sql很慢有3种原因: sql本身查询不合理,需要优化数据库没有索引多次频繁访问数据,造成了不必要的开销 取消多次获取数据,一次获取 框定一个大致的范围,获取此次查询的所有数据使用map设置数据,没有主键使用傅和主键拼接数据 /…...

NVIDIA k8s-device-plugin源码分析与安装部署

在《kubernetes Device Plugin原理与源码分析》一文中,我们从源码层面了解了kubelet侧关于device plugin逻辑的实现逻辑,本文以nvidia管理GPU的开源github项目k8s-device-plugin为例,来看看设备插件侧的实现示例。 一、Kubernetes Device Pl…...