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 中的跨程序调用

Solana 计算单元与交易费用概述

  • 0xE
  • 发布于 2025-03-26 10:22
  • 阅读 1381

本文对比了以太坊和 Solana 的交易费用与计算模型,深入探讨了 Solana 计算单元(CU)的使用、优化策略及字节码执行逻辑,并通过示例验证了 CU 消耗与费用无关的特点,同时介绍了 eBPF 和 SBF 的技术背景。

计算单元与交易费用的基本概念

在以太坊中,交易费用通过公式 gasUsed × gasPrice 计算,反映包含交易所需的以太币成本。发送交易前需预设 gasLimit,若燃气耗尽,交易回滚。

Solana 则不同,其操作码/指令消耗“计算单元”(Compute Units, CU),而非“燃气”。每个交易默认上限为 20 万 CU(可额外付费提升至 140 万 CU),超出限制则交易回滚。与以太坊将存储成本纳入燃气计算不同,Solana 的持久存储定价另行处理,因此本文聚焦操作码执行的费用模型。

两者的字节码执行逻辑相似:以太坊运行 EVM 字节码,Solana 使用改进的 伯克利数据包过滤器(即 SBF,Solana Bytecode Format),并按指令收费。以太坊根据指令复杂性收取 1 至数千燃气,Solana 则统一每条指令为 1 CU。


计算单元不足时的应对策略

对于超出限制的重型计算任务,常用策略是将工作分拆,跨多个交易执行。这需要“保存中间状态”至持久存储(后续文章详述)。类似以太坊处理大循环时,需存储“停止索引”和“计算结果”两个变量。


计算单元优化

Solana 使用 CU 限制运行时资源,防止无限循环或停机问题。交易默认上限为 20 万 CU,若超出,所有状态变更回滚,费用不退还,保护网络免受恶意密集计算攻击。

与 EVM 链不同,Solana 交易费用与 CU 消耗无关。无论使用 400 CU 还是 20 万 CU,费用恒定,仅由签名数量决定。据 Solana 文档,每个签名固定收取 5000 lamports,交易最多支持 12 个签名(受 1232 字节大小限制)。

示例验证

以下空程序展示基础费用:

use anchor_lang::prelude::*;

declare_id!("HDSz41Lh841S6AFdLBw1bt34NLkFNY5TAUsfzyFqmQKf");

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

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

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

测试代码:


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

describe("compute_unit", () => {
  anchor.setProvider(anchor.AnchorProvider.env());

  const program = anchor.workspace.ComputeUnit as Program<ComputeUnit>;

  const defaultKeyPair = new anchor.web3.PublicKey("5NhLjdFKocoRMqic9sqAe5TxLagJCoCBunzg51ioMYot");

  it("Is initialized!", async () => {

    let bal_before = await program.provider.connection.getBalance(
      defaultKeyPair
    );
    console.log("before:", bal_before);

    const tx = await program.methods.initialize().rpc();

    let bal_after = await program.provider.connection.getBalan...

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

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

0 条评论

请先 登录 后评论