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

solidity高阶 -- Eth支付

        在区块链的世界里,智能合约是实现去中心化应用(DApp)的核心技术之一。Solidity 是一种专门用于编写以太坊智能合约的编程语言,它可以帮助开发者构建各种功能,包括支付功能。

        今天,我们就来探讨如何使用 Solidity 实现以太坊主币的支付功能,以及如何通过智能合约接收和管理这些资金。

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

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

1. 以太坊支付的原理

        在以太坊网络中,支付功能的核心是通过智能合约接收和管理以太币(ETH)。Solidity 提供了一个关键字 payable,它允许一个地址或函数接收以太币。当一个函数被标记为 payable 时,用户可以通过调用该函数并发送以太币来执行合约逻辑。

1. 什么是payable

在Solidity中,payable是一个修饰符,用于标记函数或地址可以接收以太坊主币。如果一个函数被标记为payable,那么当调用该函数时,可以附带发送ETH。同样,如果一个地址被标记为payable,那么可以向该地址发送ETH。

2. 实现支付功能的代码示例

以下是一个简单的 Solidity 合约示例,展示了如何实现接收以太币的功能:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.3;contract Payable {address public owner;constructor() {owner = payable(msg.sender); // 将合约部署者设置为所有者,并确保其地址是 payable 类型}// 接收以太币的函数function deposit() external payable {}// 查询合约余额的函数function getBalance() external view returns (uint) {return address(this).balance; // 返回合约当前的以太币余额}
}

 

代码解析

  1. payable 关键字

    • payable 是 Solidity 中的一个关键字,用于标记地址或函数可以接收以太币。

    • constructor() 中,owner = payable(msg.sender) 将合约部署者的地址标记为 payable 类型,这样就可以接收以太币。

    • deposit() 函数中,payable 关键字使得该函数可以接收用户发送的以太币。

  2. msg.senderaddress(this)

    • msg.sender 是一个全局变量,表示调用当前函数的地址。

    • address(this) 表示当前合约的地址,address(this).balance 可以获取合约当前的以太币余额。

  3. externalview

    • external 表示函数只能从合约外部调用。

    • view 表示该函数不会修改区块链上的状态,只是读取数据。

 

3. 如何使用这个合约

部署合约

  1. 使用以太坊开发工具(如 Remix、Truffle 或 Hardhat)将合约部署到以太坊网络上。

  2. 部署完成后,合约会生成一个地址,你可以通过这个地址与合约交互。

发送以太币

  1. 在支持以太坊的钱包(如 MetaMask)中,找到合约地址。

  2. 调用 deposit() 函数,并在交易中附带一定数量的以太币。

  3. 交易完成后,合约会接收这些以太币,并存储在合约地址中。

查询余额

  1. 调用 getBalance() 函数,可以查询合约当前的以太币余额。

 

4. 注意事项

  1. 安全性:在处理以太币时,确保合约逻辑安全,避免漏洞导致资金损失。

  2. Gas 费用:所有以太坊交易都需要支付 Gas 费用,确保用户有足够的以太币支付这些费用。

  3. 测试环境:在将合约部署到主网之前,建议在测试网络(如 Ropsten 或 Rinkeby)上进行充分测试。

5.官方文档

发送以太币:转账、发送、调用(Solidity 代码示例)

如何发送以太币?

您可以通过以下方式将 Ether 发送到其他合约

  • transfer(2300 gas,抛出错误)
  • send(2300 gas,返回 bool)
  • call(forward all gas 或 set gas,返回 bool)

如何获得以太币?

接收以太币的合约必须至少具有以下功能之一

  • receive() external payable
  • fallback() external payable

receive()如果为空,则调用 ,否则调用。msg.datafallback()

您应该使用哪种方法?

call与 re-entrancy guard 结合使用是推荐在 2019 年 12 月之后使用的方法。

防止重入

  • 在调用其他 Contract 之前进行所有 state 更改
  • 使用 Re-entrancy Guard 修饰符
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.26;import {SafeTransferLib} from "@solady/utils/SafeTransferLib.sol";contract ReceiveEther {/*Which function is called, fallback() or receive()?send Ether|msg.data is empty?/ \yes  no/     \receive() exists?  fallback()/   \yes   no/      \receive()   fallback()*/// Function to receive Ether. msg.data must be emptyreceive() external payable {}// Fallback function is called when msg.data is not emptyfallback() external payable {}function getBalance() public view returns (uint256) {return address(this).balance;}
}contract SendEther {function sendViaTransfer(address payable _to) public payable {// This function is no longer recommended for sending Ether._to.transfer(msg.value);}function sendViaSend(address payable _to) public payable {// Send returns a boolean value indicating success or failure.// This function is not recommended for sending Ether.bool sent = _to.send(msg.value);require(sent, "Failed to send Ether");}function sendViaCall(address payable _to) public payable {// Call returns a boolean value indicating success or failure.// This is the current recommended method to use.(bool sent, bytes memory data) = _to.call{value: msg.value}("");require(sent, "Failed to send Ether");}// This is 0.36% more gas-efficient per call and is more concise as wellfunction sendViaSolady(address payable _to) public payable {// Reverts with ETHTransferFailed error// this is the most gas-efficient method to useSafeTransferLib.safeTransferETH(to, msg.value);}
}

6. 总结

通过使用payable关键字,我们可以轻松地在Solidity合约中处理ETH的支付。本文中的示例合约展示了如何接收ETH并查询合约余额。你可以在此基础上扩展功能,例如实现ETH的提现、设置支付条件等。

希望这篇博客对你理解Solidity中的ETH支付有所帮助!如果你有任何问题或建议,欢迎在评论区留言。


注意:在实际开发中,请务必注意安全性问题,尤其是在处理资金相关的逻辑时。建议在部署到主网之前,充分测试合约并在测试网上进行验证。

http://www.lryc.cn/news/529881.html

相关文章:

  • 深入理解Java中的String
  • 洛谷 P1734 最大约数和 C语言
  • Golang 执行流程分析
  • python学opencv|读取图像(五十一)使用修改图像像素点上BGR值实现图像覆盖效果
  • Flask数据的增删改查(CRUD)_flask删除数据自动更新
  • kamailio-ACC模块介绍【kamailio6.0. X】
  • 数据库对象
  • EtherCAT主站IGH-- 27 -- IGH之globals.h文件解析
  • 2025多目标优化创新路径汇总
  • 15JavaWeb——Maven高级篇
  • 使用Ollama本地化部署DeepSeek
  • 蓝桥杯刷题DAY1:前缀和
  • 【基于SprintBoot+Mybatis+Mysql】电脑商城项目之用户注册
  • MINIRAG: TOWARDS EXTREMELY SIMPLE RETRIEVAL-AUGMENTED GENERATION论文翻译
  • 微服务入门(go)
  • Baklib揭示内容中台实施最佳实践的策略与实战经验
  • C++11新特性之lambda表达式
  • 洛谷 P10289 [GESP样题 八级] 小杨的旅游 C++ 完整题解
  • 使用 Tauri 2 + Next.js 开发跨平台桌面应用实践:Singbox GUI 实践
  • JWT入门
  • Python - Quantstats量化投资策略绩效统计包 - 详解
  • 智慧园区管理系统推动企业智能运维与资源优化的全新路径分析
  • 【数据结构-字典树】力扣14. 最长公共前缀
  • 《深入浅出HTTPS​​​​​​​​​​​​​​​​​》读书笔记(31):HTTPS和TLS/SSL
  • Go学习:Go语言中if、switch、for语句与其他编程语言中相应语句的格式区别
  • L30.【LeetCode笔记】设计链表
  • java日志框架详解-Log4j2
  • C++中vector追加vector
  • 加一(66)
  • 远程连接-简化登录