小试牛刀-Solana合约账户详解

目录

一.Solana

三.账户详解

[3.1 程序账户](#3.1 程序账户)

[3.2 系统所有账户](#3.2 系统所有账户)

[3.3 程序派生账户(PDA)](#3.3 程序派生账户(PDA))

[3.4 Token账户](#3.4 Token账户)

四、相关学习文档

五、在线编辑器


Welcome to Code Block's blog

本篇文章主要介绍了

[Solana合约账户详解 ]

❤博主广交技术好友,喜欢文章的可以关注一下❤

注:该篇文章在测试环境下进行,使用测试账户进行。

一.合约账户类型

开发语言上,Solana合约使用Rust 为主要开发语言,其次是Solana合约并不像其它链那样将数据直接存到合约里,而是使用了更加独立的账户来代币转移和存储数据。按功能可以分为以下账户:

  • 程序账户
  • 系统所有账户
  • 程序派生账户(PDA)
  • Token账户

二.账户详解

2.1 程序账户

程序账户即为合约部署后生成的账户,这个账户用于存储和执行智能合约代码,即合约部署后生成的地址。这里部署地址是:Fckx9Sxf17hyauwe1nfxL9GtvruESb5kTrU2LCPLJpzm

测试截图:

2.2 系统所有账户

用户通过调用程序账户生成用于存储状态的数据的账户为系统所有账户,这个账户的所有者为当前合约。同时用户可以获得当前账户的公钥和私钥信息。使用TS创建的代码如下:

TypeScript 复制代码
    const greetingAccountKp = new web3.Keypair();
    const lamports = await pg.connection.getMinimumBalanceForRentExemption(
      GREETING_SIZE
    );
    console.log(GREETING_SIZE);
    const createGreetingAccountIx = web3.SystemProgram.createAccount({
      fromPubkey: pg.wallet.publicKey,
      lamports,
      newAccountPubkey: greetingAccountKp.publicKey,
      programId: pg.PROGRAM_ID,
      space: GREETING_SIZE,
    });
    const tx = new web3.Transaction();
    tx.add(createGreetingAccountIx);

    const txHash = await web3.sendAndConfirmTransaction(pg.connection, tx, [
      pg.wallet.keypair,
      greetingAccountKp,
    ]);
    console.log(`Use 'solana confirm -v ${txHash}' to see the logs`);

执行上述代码并成功后,这里的greetingAccountKp内的公钥和私钥即为创建账户的公钥和私钥。可以通过以下方式进行打印:

TypeScript 复制代码
    console.log("publicKey:" + greetingAccountKp.publicKey.toBase58());
    console.log("secretKey:" + greetingAccountKp.secretKey);

这个账户与普通账户的不同之处在于其用于存储数据,账户owner为当前程序账户,在合约内可以使用以下方式在合约内进行打印:

TypeScript 复制代码
    // 账户列表
    let accounts_iter = &mut accounts.iter();

    // 下一个账户
    let account = next_account_info(accounts_iter)?;
    // 打印账户拥有者
    msg!("account:{}", account.owner);

log截图:

测试截图:

2.3 程序派生账户(PDA)

在实际应用中,用户在创建账户后即需要存储账户信息,这显然是不符合实际需求的,首先这会生成大量账户浪费资源,并且不利于账户管理和维护。如实现用户锁仓功能:用户创建一个账户后,服务程序即需要存储这个钱包创建的对应账户地址。PDA账户很好的解决了这个问题。PDA的原理为让程序通过利用种子(方法名或其它)+用户钱包公钥+程序账户(合约地址),三者进行加密运算生成一个钱包地址,这个地址是唯一的,在后续运算中仍然会生成这个地址,这样即解决了账户存储和管理问题。TS代码PDA账户生成方式如下:

TypeScript 复制代码
async function getPda(programId, userPubkey) {
  const [pda, bump] = await web3.PublicKey.findProgramAddressSync(
    [Buffer.from("lock"), userPubkey.toBuffer()],
    programId
  );
  console.log("PDA:", pda.toBase58());
  return pda;
}

这里的lock为种子(种子是可变的,可使用对应合约方法名,便于管理),userPubkey为用户地址,programId为程序账户账户地址.使用如下方式进行调用(会发现每次生成地址结果为同一地址):

TypeScript 复制代码
    const pda = getPda(pg.PROGRAM_ID, greetingAccountKp.publicKey);
    console.log(pda);

测试截图:

通过截图可以看到运行多次生成的地址为相同的.

2.4 Token账户

SPL账户即为存储某个特定的代币创建的账户,该账户用于存储某个特定类型的代币,创建方式如下:

TypeScript 复制代码
async function createSPLAccount() {
    const mint = new web3.PublicKey('代币合约地址'); 
    const owner = payerAccount.publicKey; // 付款账户

    // 创建 SPL 账户
    const tokenAccount = await splToken.Token.createAccount(connection, {
        mint,
        owner,
    });

    console.log('SPL 账户创建成功:', tokenAccount.toBase58());
}

三、相关学习文档

1.Solana官方文档

2.Solana GitHub 仓库

四、在线编辑器

1.Solana IDE

声明:该文章只作为学习和使用相关,不涉及投资等其它建议.

相关推荐
FISCO_BCOS7 小时前
【区块链 + 智慧政务】广州市黄埔区企业链上服务平台 | FISCO BCOS应用案例
区块链·政务·黄埔区企业链上服务平台
MavenTalk9 小时前
Web3学习路线图,从入门到精通
学习·web3·区块链·crypto
吾名招财11 小时前
5,智能合约(react+区块链实战)
react.js·区块链·智能合约
Footprint_Analytics12 小时前
2024 年 6 月公链行业研报:市场回调,比特币和以太坊 Layer 2 表现各异
游戏·web3·区块链
FISCO_BCOS12 小时前
【区块链 + 智慧政务】城市大脑数据监管平台 | FISCO BCOS应用案例
区块链·政务·城市大脑数据监管平台
FISCO_BCOS21 小时前
【区块链 + 智慧政务】省级一体化区块链平台 | FISCO BCOS应用案例
区块链·政务·一体化区块链平台
唐伯虎点·蚊香1 天前
发行自己的ERC20代币
区块链
FISCO_BCOS1 天前
【区块链 + 智慧政务】都江堰区块链公共服务应用平台 | FISCO BCOS应用案例
区块链·政务·都江堰区块链公共服务应用平台
qiquandong1 天前
什么是平值、实值、虚值期权合约?有哪些区别?
区块链
FISCO_BCOS1 天前
【区块链+跨境服务】基于区块链的离岸贸易综合服务平台 | FISCO BCOS应用案例
区块链·离岸贸易