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

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 empty
    receive() external payable {}

    // Fallback function is called when msg.data is not empty
    fallback() 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 well
    function sendViaSolady(address payable _to) public payable {
        // Reverts with ETHTransferFailed error
        // this is the most gas-efficient method to use
        SafeTransferLib.safeTransferETH(to, msg.value);
    }
}

6. 总结

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

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


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


http://www.kler.cn/a/530035.html

相关文章:

  • 谭浩强C语言程序设计(4) 8章(下)
  • Linux网络 HTTPS 协议原理
  • Oracle数据库高效管理与优化实践
  • 记录 | 基于MaxKB的文字生成视频
  • WinDBG查找C++句柄泄露
  • mysql中in和exists的区别?
  • 快速提升网站收录:利用网站用户反馈机制
  • Python函数基础
  • 系统思考—决策
  • Python 深拷贝与浅拷贝:数据复制的奥秘及回溯算法中的应用
  • deepseek+vscode自动化测试脚本生成
  • Error: Expected a mutable image
  • C++:抽象类习题
  • 用Python替代OpenMV IDE显示openmv USB 图像
  • vscode+vue3+高得地图开发过过程中本地视频及地图json文件的发布问题
  • 算法题(55):用最少数量的箭引爆气球
  • 【贪心算法篇】:“贪心”之旅--算法练习题中的智慧与策略(二)
  • DeepSeek R1 简易指南:架构、本地部署和硬件要求
  • 软件工程概论试题五
  • Visual Basic语言的云计算
  • 设计模式-创建型模式-建造者模式
  • 【单层神经网络】基于MXNet的线性回归实现(底层实现)
  • 【华为OD-E卷 - 最大矩阵和 100分(python、java、c++、js、c)】
  • Mac上有哪些好用的开源粘贴板app
  • TB6600和DM542C两种常见的步进电机驱动器
  • 数据库安全管理中的权限控制:保护数据资产的关键措施