Solana 学习开发之旅

2025年04月03日更新 33 人订阅
原价: ¥ 20 限时优惠
专栏简介 【Solana】使用 CLI 创建 SPL 标准的 Token 以及基础使用 【Solana】完善 SPL Token 名称和 Logo 【Solana】创建 SPL 标准的 NFT 以及完善 metadata 【Solana】一些基本的js脚本 【Solana】Anchor 框架使用笔记 【Solana】Anchor 示例:通过 CPI 实现 Sol 转账与手续费收取 Solana Hello World: 安装与开发指南 Solana 与 Rust 算术入门—从 Solidity 到 Anchor Solana Anchor 程序接口定义语言(IDL) Solana Anchor 框架下的 Require 与自定义错误 Solana 程序:支持升级与无构造函数实现 Solidity 开发者必知的 Rust 语法基础 Rust 的独特语法解析 Rust 类函数宏解析 Rust 结构体、属性宏与自定义派生宏 Rust 与 Solana 中的可见性及模块化复用 Solana 中的时钟与其他区块变量 Solana 系统变量详解 Solana 日志、事件日志与历史交易查询 Solana 中的Tx.origin、msg.sender 和 onlyOwner Solana 计算单元与交易费用概述 Solana 与 Anchor 中的账户初始化 Solana 计数器教程:账户数据的读写 使用 Solana Web3.js 和 Anchor 读取账户数据 在 Solana 中实现映射表与嵌套映射表 Solana 存储成本、最大容量与账户调整 在 Anchor 中读取账户余额:Solana 的 address(account).balance Solana 中的函数修饰符与 Fallback 函数:为何不存在 Solana 中的 SOL 转移与分割:取代 msg.value 的设计 使用不同签名者修改账户:Solana 中的权限控制 PDA 与密钥对账户:Solana 中的地址与权限模型 Anchor 中的 init_if_needed 与重新初始化攻击防范 Solana 中的 Multicall:批处理交易和交易大小的限制 Solana 中的 Owner 和 Authority 删除和关闭 Solana 中的账户和程序 在 Anchor 中的 #[derive(Accounts)] 不同类型的账户 在链上读取另一个 Anchor 程序的账户数据 Anchor 中的跨程序调用

在链上读取另一个 Anchor 程序的账户数据

  • 0xE
  • 发布于 2025-04-03 10:57
  • 阅读 1491

本文展示如何在链上通过一个 Solana 程序读取另一个程序的账户数据。

在 Solidity 中,读取其他合约的存储需通过 view 函数或公共变量,而在 Solana 中,链下客户端可直接访问存储账户。本文展示如何在链上通过一个 Solana 程序读取另一个程序的账户数据。示例将涉及两个程序:data_holder 初始化并拥有数据账户,data_reader 读取其数据。


设置存储数据的 data_holder 程序

data_holder 初始化一个包含 u64 字段 x 的 PDA 账户,并赋值为 9。以下是实现代码:

Rust 代码

use anchor_lang::prelude::*;
use std::mem::size_of;

declare_id!("B2ygCyDthArX1FqVqfzo2BuBBvAd1eSkMovagAQdrH8K");

#[program]
pub mod data_holder {
    use super::*;

    pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
        let storage = &mut ctx.accounts.storage;
        storage.x = 9;
        Ok(())
    }
}

#[derive(Accounts)]
pub struct Initialize<'info> {
    #[account(
        init,
        payer = signer,
        space = size_of::<Storage>() + 8,
        seeds = [],
        bump
    )]
    pub storage: Account<'info, Storage>,
    #[account(mut)]
    pub signer: Signer<'info>,
    pub system_program: Program<'info, System>,
}

#[account]
pub struct Storage {
    pub x: u64,
}

Typescript 代码

import * as anchor from "@coral-xyz/anchor";
import { Program } from "@coral-xyz/anchor";
import { DataHolder } from "../target/types/data_holder";

describe("data-holder", () => {
  anchor.setProvider(anchor.AnchorProvider.env());
  const program = anchor.workspace.DataHolder as Program<DataHolder>;

  it("Is initialized!", async () => {
    const seeds = [];
    const [storage, _bump] = anchor.web3.PublicKey.findProgramAddressSync(seeds, program.programId);

    await program.methods.initialize().accounts({ storage: storage }).rpc();

    let storageStruct = await program.account.storage.fetch(storage);
    console.log("The value of x is: ", storageStruct.x.toString());
    console.log("Storage account address: ", storage.toBase58());
  });
});

输出

The value of x is: 9
Storage account address: 8bxHxSAfw3CRFrog3DjFytW7u4KJSc7DdGjALEE5xUWf

此 PDA 地址将用于 data_reader。


data_reader 程序

data_reader 通过 Context 获取外部账户公钥并读取数据。数据以字节形式存储,需反序列化为与 data_holder 相同的 Storage 结构体:

#[account]
pub struct Storage {
    x: u64,
}

关键读取逻辑如下:

let mut data_slice: &[u8] = &data_account.data.borrow();
let data_struct: Storage = AccountDeserialize::try_deserialize(&mut data_slice)?;

Rust 完整代码


use anchor_lang::prelude::*;

declare_id!("Bav1fsb6K3XqhZ6W2Vj4vs5G1tPyjCfLE3FPYpFrRFKn");

#[program]
pub mod data_reader {
    use super::*;

    pub fn read_other_data(ctx: Context<ReadOtherData>) -> Result<()> {
        let data_account = &ctx.accounts.other_data;

        if data_account.data_is_empty() {
            retu...

剩余50%的内容订阅专栏后可查看

点赞 1
收藏 0
分享
本文参与登链社区写作激励计划 ,好文好收益,欢迎正在阅读的你也加入。

0 条评论

请先 登录 后评论