本文详细解析了 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):
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 输出一致。
截至撰文时:
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,
}
关键参数
【笔记配套代码】 https://github.com/0xE1337/rareskills_evm_to_solana 【参考资料】 https://learnblockchain.cn/column/119 https://www.rareskills.io/solana-tutorial
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!