在 Solana 上实现 SOL 转账及构建支付分配器

与以太坊不同,在以太坊中,钱包通过 msg.value 指定交易的一部分并“推送” ETH 到合约,而 Solana 程序则是从钱包“拉取” Solana。

因此,没有“可支付”函数或“msg.value”这样的概念。

下面我们创建了一个新的 anchor 项目,名为 sol_splitter,并放置了将 SOL 从发送者转移到接收者的 Rust 代码。

当然,如果发送者直接发送 SOL,而不是通过程序来完成,这样会更高效,但我们想要说明的是如何做到这一点:

    use anchor_lang::prelude::*;use anchor_lang::system_program;declare_id!("9qnGx9FgLensJQy1hSB4b8TaRae6oWuNDveUrxoYatr7");#[program]pub mod sol_splitter {use super::*;pub fn send_sol(ctx: Context<SendSol>, amount: u64) -> Result<()> {let cpi_context = CpiContext::new(ctx.accounts.system_program.to_account_info(), system_program::Transfer {from: ctx.accounts.signer.to_account_info(),to: ctx.accounts.recipient.to_account_info(),});let res = system_program::transfer(cpi_context, amount);if res.is_ok() {return Ok(());} else {return err!(Errors::TransferFailed);}}}#[error_code]pub enum Errors {#[msg("transfer failed")]TransferFailed,}#[derive(Accounts)]pub struct SendSol<'info> {/// CHECK: we do not read or write the data of this account#[account(mut)]recipient: UncheckedAccount<'info>,system_program: Program<'info, System>,#[account(mut)]signer: Signer<'info>,}

这里有很多内容需要解释。

引入 CPI:跨程序调用

在以太坊中,转移 ETH 只需在 msg.value 字段中指定一个值。在 Solana 中,一个名为 system program 的内置程序将 SOL 从一个账户转移到另一个账户。这就是为什么在我们初始化账户时,它不断出现并且需要支付费用来初始化它们。

你可以大致将系统程序视为以太坊中的预编译。想象一下,它的行为有点像内置于协议中的 ERC-20 代币,用作原生货币。它有一个名为 transfer 的公共函数。

CPI 交易的上下文

每当调用 Solana 程序函数时,必须提供一个 Context。该 Context 包含程序将要交互的所有账户。

调用系统程序没有什么不同。系统程序需要一个 Context,其中包含 from 和 to 账户。转移的 amount 作为“常规”参数传递——它不是 Context 的一部分(因为“amount”不是一个账户,它只是一个值)。

现在我们可以解释下面的代码片段:

cpi_context system_program transfer

我们正在构建一个新的 CpiContext,它将我们要调用的程序作为第一个参数(绿色框),以及将作为该交易一部分的账户(黄色框)。参数 amount 在这里没有提供,因为 amount 不是一个账户。

现在我们已经构建了 cpi_context,可以在指定金额的同时对系统程序进行跨程序调用(橙色框)。

这返回一个 Result<()> 类型,就像我们 Anchor 程序上的公共函数一样。

不要忽视跨程序调用的返回值。

要检查跨程序调用是否成功,我们只需检查返回值是否为 Ok。Rust 通过 is_ok() 方法使这变得简单:

            let res = system_program::transfer(cpi_context, amount);if res.is_ok() {return Ok(());} else {return err!(Errors::TransferFailed);}}}#[error_code]pub enum Errors {#[msg("transfer failed")]TransferFailed,}

只有签名者可以是“from”

如果你调用系统程序时 from 是一个不是 Signer 的账户,那么系统程序将拒绝该调用。没有签名,系统程序无法知道你是否授权了该调用。

TypeScript 代码:

    import * as anchor from "@coral-xyz/anchor";import { Program } from "@coral-xyz/anchor";import { SolSplitter } from "../target/types/sol_splitter";describe("sol_splitter", () => {// Configure the client to use the local cluster.anchor.setProvider(anchor.AnchorProvider.env());const program = anchor.workspace.SolSplitter as Program<SolSplitter>;async function printAccountBalance(account) {const balance = await anchor.getProvider().connection.getBalance(account);console.log(`${account} has ${balance / anchor.web3.LAMPORTS_PER_SOL} SOL`);}it("Transmit SOL", async () => {// generate a new walletconst recipient = anchor.web3.Keypair.generate();await printAccountBalance(recipient.publicKey);// send the account 1 SOL via the programlet amount = new anchor.BN(1 * anchor.web3.LAMPORTS_PER_SOL);await program.methods.sendSol(amount).accounts({recipient: recipient.publicKey}).rpc();await printAccountBalance(recipient.publicKey);});});

一些需要注意的事项:

  • 我们创建了一个辅助函数 printAccountBalance 来显示余额的前后
  • 我们使用 anchor.web3.Keypair.generate() 生成了接收者钱包
  • 我们将 1 SOL 转移到新账户

当我们运行代码时,预期结果如下。打印语句是接收者地址的前后余额:

result sol_sprinter

练习:构建一个 Solana 程序,将传入的 SOL 平均分配给两个接收者。你将无法通过函数参数来完成此操作,账户需要在 Context 结构中。

构建支付分割器:使用 remaining_accounts 处理任意数量的账户。

我们可以看到,如果我们想将 SOL 分配给多个账户,必须指定一个 Context 结构会显得相当笨拙:

    #[derive(Accounts)]pub struct SendSol<'info> {/// CHECK: we do not read or write the data of this account#[account(mut)]recipient1: UncheckedAccount<'info>,/// CHECK: we do not read or write the data of this account#[account(mut)]recipient2: UncheckedAccount<'info>,/// CHECK: we do not read or write the data of this account#[account(mut)]recipient3: UncheckedAccount<'info>,// .../// CHECK: we do not read or write the data of this account#[account(mut)]recipientn: UncheckedAccount<'info>,system_program: Program<'info, System>,#[account(mut)]signer: Signer<'info>,}

为了解决这个问题,Anchor 在 Context 结构中添加了一个 remaining_accounts 字段。

下面的代码说明了如何使用这个特性:

    use anchor_lang::prelude::*;use anchor_lang::system_program;declare_id!("9qnGx9FgLensJQy1hSB4b8TaRae6oWuNDveUrxoYatr7");#[program]pub mod sol_splitter {use super::*;// 'a, 'b, 'c 是 Rust 生命周期,暂时忽略它们pub fn split_sol<'a, 'b, 'c, 'info>(ctx: Context<'a, 'b, 'c, 'info, SplitSol<'info>>,amount: u64,) -> Result<()> {let amount_each_gets = amount / ctx.remaining_accounts.len() as u64;let system_program = &ctx.accounts.system_program;// 注意关键字 `remaining_accounts`for recipient in ctx.remaining_accounts {let cpi_accounts = system_program::Transfer {from: ctx.accounts.signer.to_account_info(),to: recipient.to_account_info(),};let cpi_program = system_program.to_account_info();let cpi_context = CpiContext::new(cpi_program, cpi_accounts);let res = system_program::transfer(cpi_context, amount_each_gets);if !res.is_ok() {return err!(Errors::TransferFailed);}}Ok(())}}#[error_code]pub enum Errors {#[msg("transfer failed")]TransferFailed,}#[derive(Accounts)]pub struct SplitSol<'info> {#[account(mut)]signer: Signer<'info>,system_program: Program<'info, System>,}

这是 TypeScript 代码:

    import * as anchor from "@coral-xyz/anchor";import { Program } from "@coral-xyz/anchor";import { SolSplitter } from "../target/types/sol_splitter";describe("sol_splitter", () => {// 配置客户端以使用本地集群。anchor.setProvider(anchor.AnchorProvider.env());const program = anchor.workspace.SolSplitter as Program<SolSplitter>;async function printAccountBalance(account) {const balance = await anchor.getProvider().connection.getBalance(account);console.log(`${account} has ${balance / anchor.web3.LAMPORTS_PER_SOL} SOL`);}it("Split SOL", async () => {const recipient1 = anchor.web3.Keypair.generate();const recipient2 = anchor.web3.Keypair.generate();const recipient3 = anchor.web3.Keypair.generate();await printAccountBalance(recipient1.publicKey);await printAccountBalance(recipient2.publicKey);await printAccountBalance(recipient3.publicKey);const accountMeta1 = {pubkey: recipient1.publicKey, isWritable: true, isSigner: false};const accountMeta2 = {pubkey: recipient2.publicKey, isWritable: true, isSigner: false};const accountMeta3 = {pubkey: recipient3.publicKey, isWritable: true, isSigner: false};let amount = new anchor.BN(1 * anchor.web3.LAMPORTS_PER_SOL);await program.methods.splitSol(amount).remainingAccounts([accountMeta1, accountMeta2, accountMeta3]).rpc();await printAccountBalance(recipient1.publicKey);await printAccountBalance(recipient2.publicKey);await printAccountBalance(recipient3.publicKey);});});

运行测试显示了之前和之后的余额:

test result Split SOL

这里是对 Rust 代码的一些评论:

Rust 生命周期

split_sol 的函数声明有一些奇怪的语法:

    pub fn split_sol<'a, 'b, 'c, 'info>(ctx: Context<'a, 'b, 'c, 'info, SplitSol<'info>>,amount: u64,) -> Result<()>

'a'b 和 'c 是 Rust 生命周期。Rust 生命周期是一个复杂的话题,我们暂时不想涉及。但简单来说,Rust 代码需要确保在循环 for recipient in ctx.remaining_accounts 中传入的资源在整个循环期间始终存在。

ctx.remaining_accounts

循环通过 for recipient in ctx.remaining_accounts 遍历。关键字 remaining_accounts 是 Anchor 机制,用于传递任意数量的账户,而不必在 Context 结构中创建一堆键。

在 TypeScript 测试中,我们可以像这样将 remaining_accounts 添加到事务中:

    await program.methods.splitSol(amount).remainingAccounts([accountMeta1, accountMeta2, accountMeta3]).rpc();

 更多相关信息,请,https://t.me/gtokentool

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

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

相关文章

WebRTC搭建与应用(一)-ICE服务搭建

WebRTC搭建与应用(一) 近期由于项目需要在研究前端WebGL渲染转为云渲染&#xff0c;借此机会对WebRTC、ICE信令协议等有了初步了解&#xff0c;在此记录一下&#xff0c;以防遗忘。 第一章 ICE服务搭建 文章目录 WebRTC搭建与应用(一)前言一、ICE是什么&#xff1f;二、什么…

Linux高性能服务器编程 | 读书笔记 | 12. 多线程编程

12. 多线程编程 注&#xff1a;博客中有书中没有的内容&#xff0c;均是来自 黑马06-线程概念_哔哩哔哩_bilibili 早期Linux不支持线程&#xff0c;直到1996年&#xff0c;Xavier Leroy等人开发出第一个基本符合POSIX标准的线程库LinuxThreads&#xff0c;但LinuxThreads效率…

查看Mysql数据库引擎以及修改引擎为innoDB

目录 打开Mysql命令行 打开Mysql命令行 SHOW ENGINES;innoDB在事务型数据库中应用最多&#xff0c;其主要支持事务安全表&#xff08;ACID&#xff09;&#xff0c;行锁定和外键。 介绍下InnoDB的主要特性&#xff1a; 1、InnoDB给MySQL提供了具有提交、回滚和崩溃恢复能力的事…

Moretl安全日志采集工具

永久免费: 至Gitee下载 使用教程: Moretl使用说明 使用咨询: 用途 定时全量或增量采集工控机,电脑文件或日志. 优势 开箱即用: 解压直接运行.不需额外下载.管理设备: 后台统一管理客户端.无人值守: 客户端自启动,自更新.稳定安全: 架构简单,兼容性好,通过授权控制访问. 架…

密码学——密码学概述、分类、加密技术(山东省大数据职称考试)

大数据分析应用-初级 第一部分 基础知识 一、大数据法律法规、政策文件、相关标准 二、计算机基础知识 三、信息化基础知识 四、密码学 五、大数据安全 六、数据库系统 七、数据仓库. 第二部分 专业知识 一、大数据技术与应用 二、大数据分析模型 三、数据科学 密码学 大数据…

nodejs搭配express网站开发后端接口设计需要注意事项

nodejs搭配express网站开发后端接口设计需要注意事项&#xff01;为了回避一些常见的误区&#xff0c;今天和大家汇总一下&#xff0c;最近我遇到的一些错误信息&#xff0c;虽然都是小问题&#xff0c;但是还是需要分享一下&#xff0c;以免大家再次犯错。 1&#xff1a;第一个…

8_HTML5 SVG (4) --[HTML5 API 学习之旅]

8_HTML5 SVG (4) --[HTML5 API 学习之旅] SVG 文本 HTML5 中的 SVG&#xff08;可缩放矢量图形&#xff09;允许你直接在网页中嵌入图形&#xff0c;并且可以使用 <text> 元素来添加文本到这些图形中。以下是四个带有详细注释的 SVG 文本示例&#xff0c;展示了如何在不…

【中标麒麟服务器操作系统实例分享】java应用DNS解析异常分析及处理

了解更多银河麒麟操作系统全新产品&#xff0c;请点击访问 麒麟软件产品专区&#xff1a;https://product.kylinos.cn 开发者专区&#xff1a;https://developer.kylinos.cn 文档中心&#xff1a;https://documentkylinos.cn 情况描述 中标麒麟服务器操作系统V7运行在 ARM虚…

谷歌浏览器的界面调整与设置方法

谷歌浏览器是一款广受欢迎的网络浏览器&#xff0c;其简洁的界面和丰富的扩展功能吸引了大量用户。本文将详细介绍如何调整谷歌浏览器的界面以及一些实用的设置方法&#xff0c;帮助你更好地使用这款浏览器。&#xff08;本文由https://chrome.sungyun.cn/的作者进行编写&#…

05、GC基础知识

JVM程序在跑起来之后&#xff0c;在数据的交互过程中&#xff0c;就会有一些数据是过期不用的&#xff0c;这些数据可以看做是垃圾&#xff0c;JVM中&#xff0c;这些垃圾是不用开发者管的&#xff0c;它自己会有一套垃圾回收系统自动回收这些内存垃圾&#xff0c;以备后面继续…

苍穹外卖-day05redis 缓存的学习

苍穹外卖-day05 课程内容 Redis入门Redis数据类型Redis常用命令在Java中操作Redis店铺营业状态设置 学习目标 了解Redis的作用和安装过程 掌握Redis常用的数据类型 掌握Redis常用命令的使用 能够使用Spring Data Redis相关API操作Redis 能够开发店铺营业状态功能代码 功能实…

Spark-Streaming集成Kafka

Spark Streaming集成Kafka是生产上最多的方式&#xff0c;其中集成Kafka 0.10是较为简单的&#xff0c;即&#xff1a;Kafka分区和Spark分区之间是1:1的对应关系&#xff0c;以及对偏移量和元数据的访问。与高版本的Kafka Consumer API 集成时做了一些调整&#xff0c;下面我们…

启动报错java.lang.NoClassDefFoundError: ch/qos/logback/core/status/WarnStatus

报错信息图片 日志&#xff1a; Exception in thread "Quartz Scheduler [scheduler]" java.lang.NoClassDefFoundError: ch/qos/logback/core/status/WarnStatus先说我自己遇到的问题&#xff0c;我们项目在web设置了自定义的log输出路径&#xff0c;多了一个 / 去…

2025erp系统开源免费进销存系统搭建教程/功能介绍/上线即可运营软件平台源码

系统介绍 基于ThinkPHP与LayUI构建的全方位进销存解决方案 本系统集成了采购、销售、零售、多仓库管理、财务管理等核心功能模块&#xff0c;旨在为企业提供一站式进销存管理体验。借助详尽的报表分析和灵活的设置选项&#xff0c;企业可实现精细化管理&#xff0c;提升运营效…

数据增强的几大方式

1. 随机擦除(Random Erasing) 说明 随机在图像中选取一个矩形区域&#xff0c;将其像素值随机化或设为零&#xff0c;以增加模型对部分缺失信息的鲁棒性。 import numpy as np import cv2def random_erasing(image, sl0.02, sh0.2, r10.3):h, w, _ image.shapearea h * wta…

leecode416.分割等和子集

这道题目看了题解把题目转化为01背包问题才恍然大悟&#xff0c;sum为数组的总和&#xff0c;背包容量为sum/2&#xff0c;价值和背包重量都为nums[i]&#xff0c;由于价值和背包重量都为nums[i]&#xff0c;那么容量为sum/2的背包最多只能获得最大的价值是sum/2&#xff0c;所…

首次下载steam更新速度慢解决方法

下载免费的加速器&#xff0c;在加速器的steam商店加速页面→加速后页面上方的区服选择 &#xff08;香港移动&#xff09;→双箭头→改为登录异常专用→在下部的登录修复进入steam更新 就好了&#xff0c;亲测有效

芯片级IO (Pad) Ring IP Checklist

SoC top顶层数字后端实现都会涉及到IO Ring &#xff08;PAD Ring&#xff09;的设计。这里面包括VDD IO,VDDIO IO, Signal IO, Corner IO&#xff0c;Filler IO&#xff0c;IO power cut cell等等。 数字后端零基础入门系列 | Innovus零基础LAB学习Day2 数字IC后端实现TOP F…

通过算法识别运行过程中产生的常见缺陷,及时处理,避免运行故障,影响正常作业的智慧快消开源了

智慧快消视频监控平台是一款功能强大且简单易用的实时算法视频监控系统。它的愿景是最底层打通各大芯片厂商相互间的壁垒&#xff0c;省去繁琐重复的适配流程&#xff0c;实现芯片、算法、应用的全流程组合&#xff0c;从而大大减少企业级应用约95%的开发成本。 基于多年的深度…

μC/OS-Ⅱ源码学习(7)---软件定时器

快速回顾 μC/OS-Ⅱ中的多任务 μC/OS-Ⅱ源码学习(1)---多任务系统的实现 μC/OS-Ⅱ源码学习(2)---多任务系统的实现(下) μC/OS-Ⅱ源码学习(3)---事件模型 μC/OS-Ⅱ源码学习(4)---信号量 μC/OS-Ⅱ源码学习(5)---消息队列 μC/OS-Ⅱ源码学习(6)---事件标志组 本文进一…