Solana 存储成本、最大容量与账户调整

  • 0xE
  • 发布于 3天前
  • 阅读 307

本文详细解析了 Solana 的存储成本计算、租金免租机制、账户大小限制及动态调整方法,结合 Rust 示例展示了如何初始化和扩展账户,并与以太坊成本进行了对比。

存储成本与租金机制

在 Solana 中,创建账户时,付款人需为每字节存储空间支付一定 SOL,称为“租金”(rent)。此名称易生误解,似指定期续费,实则不然。一旦支付相当于两年租金的费用,账户即获“免租”(rent-exempt)状态,无需后续缴费。

“租金”概念源自 Solana 早期按年计费的设计:支付不足两年的账户会在期限后被回收;支付两年费用则永久免租。现今,所有账户必须初始即达免租标准,不允许支付低于两年的租金。

即使是零数据账户也有成本,因 Solana 需维护其索引和元数据(如所有者、公钥等)。初始化时,Anchor 自动计算所需租金,无需手动介入,但了解成本估算有助于应用设计。

租金计算示例

使用命令行快速估算:

solana rent 32
# 输出: Rent-exempt minimum: 0.0011136 SOL

solana rent 0
# 输出: Rent-exempt minimum: 0.00089088 SOL

计算公式

Anchor 提供租金相关常量(位于 solana_program::rent):

  • ACCOUNT_STORAGE_OVERHEAD:128 字节,账户元数据开销。
  • DEFAULT_EXEMPTION_THRESHOLD:2.0,两年免租阈值。
  • DEFAULT_LAMPORTS_PER_BYTE_YEAR:3480 lamports/字节/年,两年为 6960 lamports/字节。

Rust 示例计算空账户成本:

use anchor_lang::prelude::*;
use anchor_lang::solana_program::rent as rent_module;

declare_id!("5SiSojgeQbznoSUMk4rBzzVBRtiTt54TeCCG1Xh7WmrR");

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

    pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
        let cost = rent_module::ACCOUNT_STORAGE_OVERHEAD as f64
            * rent_module::DEFAULT_LAMPORTS_PER_BYTE_YEAR as f64
            * rent_module::DEFAULT_EXEMPTION_THRESHOLD;
        msg!("cost to create an empty account: {}", cost); // 890880
        Ok(())
    }
}

#[derive(Accounts)]
pub struct Initialize {}

扩展至 32 字节账户:

pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
    let base_cost = rent_module::ACCOUNT_STORAGE_OVERHEAD as f64
        * rent_module::DEFAULT_LAMPORTS_PER_BYTE_YEAR as f64
        * rent_module::DEFAULT_EXEMPTION_THRESHOLD;
    msg!("cost to create an empty account: {}", base_cost); // 890880

    let total_cost = base_cost + 32.0 * rent_module::DEFAULT_LAMPORTS_PER_BYTE_YEAR as f64
        * rent_module::DEFAULT_EXEMPTION_THRESHOLD;
    msg!("cost to create a 32 byte account: {}", total_cost); // 1113600
    Ok(())
}

结果与 solana rent 输出一致。

与以太坊比较

截至撰文时:

  • 以太坊:ETH ≈ $2425,初始化账户 22,100 gas,按 15 gwei/gas 计算,约 $0.80。
  • Solana:SOL ≈ $90,32 字节账户 1,113,600 lamports,约 $0.10。
  • 若 SOL 市值达 ETH 水平(约 $675),成本升至 $0.75。

Solana 通胀率最终稳定于 1.5%/年,反映存储成本随技术进步降低,但常量可通过硬分叉调整。


余额不足免租的后果

若账户余额低于两年免租阈值,Solana 运行时将逐步扣减余额支付租金,直至账户被回收。相关讨论见 Reddit 帖子。回收后可通过充值 SOL 复活账户,但数据将丢失。


账户大小限制

初始化账户时,最大分配为 10,240 字节(10 KB)。账户总上限为 10 MB,通过后续调整实现。


调整账户大小

使用 realloc 修饰符动态调整账户大小,常用于扩展向量等动态数据结构。以下示例将账户增加 1000 字节:

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

declare_id!("9XZGDi1imvGFV3bLuDJK1bkUht51GPCRsVxMe4DpqWEo");

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

    pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
        Ok(())
    }

    pub fn increase_account_size(ctx: Context<IncreaseAccountSize>) -> Result<()> {
        Ok(())
    }
}

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

#[derive(Accounts)]
pub struct IncreaseAccountSize<'info> {
    #[account(
        mut,
        realloc = size_of::<MyStorage>() + 8 + 1000, // 增加 1000 字节
        realloc::payer = signer,
        realloc::zero = false,
        seeds = [],
        bump
    )]
    pub my_storage: Account<'info, MyStorage>,
    #[account(mut)]
    pub signer: Signer<'info>,
    pub system_program: Program<'info, System>,
}

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

关键参数

  • realloc:指定新大小。
  • realloc::payer:支付调整费用的账户。
  • realloc::zero:false 保留现有数据,true 清零。

账户最大容量

  • 单次 realloc 增量上限:10,240 字节。
  • 账户总上限:10 MB。

【笔记配套代码】 https://github.com/0xE1337/rareskills_evm_to_solana 【参考资料】 https://learnblockchain.cn/column/119 https://www.rareskills.io/solana-tutorial

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

0 条评论

请先 登录 后评论
0xE
0xE
0x59f6...a17e
17年进入币圈,Web3 开发者。刨根问底探链上真相,品味坎坷悟 Web3 人生。