Tapahtuman rakenne

Yhteenveto

Tapahtuma sisältää allekirjoitukset + viestin. Viesti sisältää otsikon, tiliosoitteet, viimeaikaisen lohkohajautusarvon ja käännetyt ohjeet. Sarjallistetun koon maksimi: 1 232 tavua.

Transaction-rakenteella on kaksi ylätason kenttää:

  • signatures: Allekirjoitusten taulukko
  • message: Tapahtumatiedot, mukaan lukien luettelo käsiteltävistä ohjeista
Transaction
pub struct Transaction {
pub signatures: Vec<Signature>,
pub message: Message,
}

Kaavio, joka näyttää tapahtuman kaksi osaaKaavio, joka näyttää tapahtuman kaksi osaa

Tapahtuman sarjallistetun kokonaiskoon ei saa ylittää PACKET_DATA_SIZE-arvoa (1 232 tavua). Tämä rajoitus vastaa 1 280 tavua (IPv6:n vähimmäis-MTU) miinus 48 tavua verkon otsikoille (40 tavua IPv6 + 8 tavua fragmenttiotsikko). 1 232 tavua sisältää sekä signatures-taulukon että message-rakenteen.

Kaavio, joka näyttää tapahtumamuodon ja kokorajoituksetKaavio, joka näyttää tapahtumamuodon ja kokorajoitukset

Allekirjoitukset

signatures-kenttä on kompaktisti koodattu taulukko Signature-arvoja. Jokainen Signature on 64-tavuinen Ed25519-allekirjoitus sarjallistetusta Message-rakenteesta, allekirjoitettu allekirjoittajatilin yksityisellä avaimella. Yksi allekirjoitus vaaditaan jokaiselta allekirjoittajatililtä, johon tapahtuman ohjeet viittaavat.

Taulukon ensimmäinen allekirjoitus kuuluu maksajalle, tilille, joka maksaa tapahtuman perusmaksun ja priorisointimaksun. Tämä ensimmäinen allekirjoitus toimii myös tapahtumatunnisteena, jota käytetään tapahtuman hakemiseen verkosta. Tapahtumatunnistetta kutsutaan yleisesti tapahtuman allekirjoitukseksi.

Maksajan vaatimukset:

  • Täytyy olla ensimmäinen tili viestissä (indeksi 0) ja allekirjoittaja.
  • Täytyy olla System Program -omistuksessa oleva tili tai nonce-tili (validoitu validate_fee_payer).
  • Täytyy sisältää riittävästi lamportteja kattamaan rent_exempt_minimum + total_fee; muuten transaktio epäonnistuu virheellä InsufficientFundsForFee.

Viesti

message-kenttä on Message -rakenne, joka sisältää transaktion hyötykuorman:

Message
pub struct Message {
/// The message header, identifying signed and read-only `account_keys`.
pub header: MessageHeader,
/// All the account keys used by this transaction.
#[serde(with = "short_vec")]
pub account_keys: Vec<Pubkey>,
/// The id of a recent ledger entry.
pub recent_blockhash: Hash,
/// Programs that will be executed in sequence and committed in
/// one atomic transaction if all succeed.
#[serde(with = "short_vec")]
pub instructions: Vec<CompiledInstruction>,
}

Otsikko

header-kenttä on MessageHeader -rakenne, jossa on kolme u8-kenttää, jotka jakavat account_keys-taulukon käyttöoikeusryhmiin:

  • num_required_signatures: Transaktion vaatimien allekirjoitusten kokonaismäärä.
  • num_readonly_signed_accounts: Allekirjoitettujen tilien määrä, jotka ovat vain luku -tilassa.
  • num_readonly_unsigned_accounts: Allekirjoittamattomien tilien määrä, jotka ovat vain luku -tilassa.
MessageHeader
pub struct MessageHeader {
/// The number of signatures required for this message to be considered
/// valid. The signers of those signatures must match the first
/// `num_required_signatures` of [`Message::account_keys`].
pub num_required_signatures: u8,
/// The last `num_readonly_signed_accounts` of the signed keys are read-only
/// accounts.
pub num_readonly_signed_accounts: u8,
/// The last `num_readonly_unsigned_accounts` of the unsigned keys are
/// read-only accounts.
pub num_readonly_unsigned_accounts: u8,
}

Kaavio, joka näyttää viestin otsikon kolme osaaKaavio, joka näyttää viestin otsikon kolme osaa

Tiliosoitteet

account_keys -kenttä on kompaktisti koodattu taulukko julkisista avaimista. Jokainen merkintä tunnistaa tilin, jota vähintään yksi transaktion käskyistä käyttää. Taulukon täytyy sisältää jokainen tili ja sen täytyy noudattaa tätä tarkkaa järjestystä:

  1. Allekirjoittaja + kirjoitettava
  2. Allekirjoittaja + vain luku
  3. Ei-allekirjoittaja + kirjoitettava
  4. Ei-allekirjoittaja + vain luku

Tämä tarkka järjestys mahdollistaa sen, että account_keys-taulukko voidaan yhdistää viestin header-kentän kolmeen lukumäärään, jolloin kunkin tilin käyttöoikeudet voidaan määrittää ilman tilikohtaisten metatietolippujen tallentamista. Otsikon lukumäärät jakavat taulukon yllä lueteltuihin neljään käyttöoikeusryhmään.

Kaavio, joka näyttää tilisoitteiden taulukon järjestyksenKaavio, joka näyttää tilisoitteiden taulukon järjestyksen

Viimeaikainen lohkohajautusarvo

recent_blockhash-kenttä on 32-tavuinen hajautusarvo, joka palvelee kahta tarkoitusta:

  1. Aikaleima: todistaa, että transaktio on luotu äskettäin.
  2. Deduplikointi: estää saman transaktion käsittelyn kahdesti.

Lohkohajautusarvo vanhenee 150 slotin jälkeen. Jos lohkohajautusarvo ei ole enää voimassa transaktion saapuessa, se hylätään virheellä BlockhashNotFound, ellei kyseessä ole voimassa oleva kestävä nonce-transaktio.

getLatestBlockhash RPC-metodi mahdollistaa nykyisen lohkohajautusarvon ja viimeisen lohkokorkeuden hakemisen, jolla lohkohajautusarvo on voimassa.

Käskyt

instructions-kenttä on kompaktisti koodattu taulukko CompiledInstruction-rakenteita. Jokainen CompiledInstruction viittaa tileihin indeksillä account_keys-taulukkoon täyden julkisen avaimen sijaan. Se sisältää:

  1. program_id_index: Indeksi account_keys-taulukkoon, joka tunnistaa kutsuttavan ohjelman.
  2. accounts: Taulukko indeksejä account_keys-taulukkoon, jotka määrittävät ohjelmalle välitettävät tilit.
  3. data: Tavutaulukko, joka sisältää käskyn erottelijan ja sarjallistetut argumentit.
CompiledInstruction
pub struct CompiledInstruction {
/// Index into the transaction keys array indicating the program account that executes this instruction.
pub program_id_index: u8,
/// Ordered indices into the transaction keys array indicating which accounts to pass to the program.
#[serde(with = "short_vec")]
pub accounts: Vec<u8>,
/// The program input data.
#[serde(with = "short_vec")]
pub data: Vec<u8>,
}

Kompakti käskyjen taulukkoKompakti käskyjen taulukko

Transaktion binäärimuoto

Transaktiot sarjallistetaan käyttäen kompaktia koodausjärjestelmää. Kaikki muuttuvan pituiset taulukot (allekirjoitukset, tiliavaimet, käskyt) alkavat compact-u16-pituuskoodauksella. Tämä muoto käyttää 1 tavua arvoille 0-127 ja 2-3 tavua suuremmille arvoille.

Legacy-transaktion rakenne (verkossa):

KenttäKokoKuvaus
num_signatures1-3 tavua (compact-u16)Allekirjoitusten määrä
signaturesnum_signatures x 64 tavuaEd25519-allekirjoitukset
num_required_signatures1 tavuMessageHeader-kenttä 1
num_readonly_signed1 tavuMessageHeader-kenttä 2
num_readonly_unsigned1 tavuMessageHeader-kenttä 3
num_account_keys1-3 tavua (compact-u16)Staattisten tiliavainten määrä
account_keysnum_account_keys x 32 tavuaJulkiset avaimet
recent_blockhash32 tavuaLohkohajautusarvo
num_instructions1-3 tavua (compact-u16)Käskyjen määrä
instructionsvaihtelevaTaulukko käännettyjä käskyjä

Jokainen käännetty ohje sarjallistetaan seuraavasti:

KenttäKokoKuvaus
program_id_index1 tavuIndeksi tiliavaimiin
num_accounts1-3 tavua (compact-u16)Tili-indeksien määrä
account_indicesnum_accounts x 1 tavuTiliavainten indeksit
data_len1-3 tavua (compact-u16)Instruction datan pituus
datadata_len tavuaLäpinäkymätön instruction data

Koon laskeminen

Kun PACKET_DATA_SIZE = 1 232 tavua, käytettävissä oleva tila voidaan laskea:

Total = 1232 bytes
- compact-u16(num_sigs) # 1 byte
- num_sigs * 64 # signature bytes
- 3 # message header
- compact-u16(num_keys) # 1 byte
- num_keys * 32 # account key bytes
- 32 # recent blockhash
- compact-u16(num_ixs) # 1 byte
- sum(instruction_sizes) # per-instruction overhead + data

Esimerkki: SOL-siirtotransaktio

Alla oleva kaavio näyttää, miten transaktiot ja ohjeet toimivat yhdessä mahdollistaen käyttäjien vuorovaikutuksen verkon kanssa. Tässä esimerkissä SOL siirretään yhdeltä tililtä toiselle.

Lähettäjätilin metadata osoittaa, että sen on allekirjoitettava transaktio. Tämä mahdollistaa System Programin vähentää lamportteja. Sekä lähettäjän että vastaanottajan tilien on oltava kirjoitettavia, jotta niiden lamportti-saldo voi muuttua. Tämän ohjeen suorittamiseksi lähettäjän lompakko lähettää transaktion, joka sisältää sen allekirjoituksen ja viestin, joka sisältää SOL-siirto-ohjeen.

SOL-siirtokaavioSOL-siirtokaavio

Kun transaktio on lähetetty, System Program käsittelee siirto-ohjeen ja päivittää molempien tilien lamportti-saldon.

SOL-siirtoprosessikaavioSOL-siirtoprosessikaavio

Alla oleva esimerkki näyttää yllä oleviin kaavioihin liittyvän koodin. Katso System Programin transfer-funktio.

import {
airdropFactory,
appendTransactionMessageInstructions,
createSolanaRpc,
createSolanaRpcSubscriptions,
createTransactionMessage,
generateKeyPairSigner,
getSignatureFromTransaction,
lamports,
pipe,
sendAndConfirmTransactionFactory,
setTransactionMessageFeePayerSigner,
setTransactionMessageLifetimeUsingBlockhash,
signTransactionMessageWithSigners
} from "@solana/kit";
import { getTransferSolInstruction } from "@solana-program/system";
// Create a connection to cluster
const rpc = createSolanaRpc("http://localhost:8899");
const rpcSubscriptions = createSolanaRpcSubscriptions("ws://localhost:8900");
// Generate sender and recipient keypairs
const sender = await generateKeyPairSigner();
const recipient = await generateKeyPairSigner();
const LAMPORTS_PER_SOL = 1_000_000_000n;
const transferAmount = lamports(LAMPORTS_PER_SOL / 100n); // 0.01 SOL
// Fund sender with airdrop
await airdropFactory({ rpc, rpcSubscriptions })({
recipientAddress: sender.address,
lamports: lamports(LAMPORTS_PER_SOL), // 1 SOL
commitment: "confirmed"
});
// Check balance before transfer
const { value: preBalance1 } = await rpc.getBalance(sender.address).send();
const { value: preBalance2 } = await rpc.getBalance(recipient.address).send();
// Create a transfer instruction for transferring SOL from sender to recipient
const transferInstruction = getTransferSolInstruction({
source: sender,
destination: recipient.address,
amount: transferAmount // 0.01 SOL in lamports
});
// Add the transfer instruction to a new transaction
const { value: latestBlockhash } = await rpc.getLatestBlockhash().send();
const transactionMessage = pipe(
createTransactionMessage({ version: 0 }),
(tx) => setTransactionMessageFeePayerSigner(sender, tx),
(tx) => setTransactionMessageLifetimeUsingBlockhash(latestBlockhash, tx),
(tx) => appendTransactionMessageInstructions([transferInstruction], tx)
);
// Send the transaction to the network
const signedTransaction =
await signTransactionMessageWithSigners(transactionMessage);
await sendAndConfirmTransactionFactory({ rpc, rpcSubscriptions })(
signedTransaction,
{ commitment: "confirmed" }
);
const transactionSignature = getSignatureFromTransaction(signedTransaction);
// Check balance after transfer
const { value: postBalance1 } = await rpc.getBalance(sender.address).send();
const { value: postBalance2 } = await rpc.getBalance(recipient.address).send();
console.log(
"Sender prebalance:",
Number(preBalance1) / Number(LAMPORTS_PER_SOL)
);
console.log(
"Recipient prebalance:",
Number(preBalance2) / Number(LAMPORTS_PER_SOL)
);
console.log(
"Sender postbalance:",
Number(postBalance1) / Number(LAMPORTS_PER_SOL)
);
console.log(
"Recipient postbalance:",
Number(postBalance2) / Number(LAMPORTS_PER_SOL)
);
console.log("Transaction Signature:", transactionSignature);
Console
Click to execute the code.

Seuraava esimerkki näyttää transaktion rakenteen, joka sisältää yhden SOL-siirto-ohjeen.

import {
createSolanaRpc,
generateKeyPairSigner,
lamports,
createTransactionMessage,
setTransactionMessageFeePayerSigner,
setTransactionMessageLifetimeUsingBlockhash,
appendTransactionMessageInstructions,
pipe,
signTransactionMessageWithSigners,
getCompiledTransactionMessageDecoder
} from "@solana/kit";
import { getTransferSolInstruction } from "@solana-program/system";
const rpc = createSolanaRpc("http://localhost:8899");
const { value: latestBlockhash } = await rpc.getLatestBlockhash().send();
// Generate sender and recipient keypairs
const sender = await generateKeyPairSigner();
const recipient = await generateKeyPairSigner();
// Define the amount to transfer
const LAMPORTS_PER_SOL = 1_000_000_000n;
const transferAmount = lamports(LAMPORTS_PER_SOL / 100n); // 0.01 SOL
// Create a transfer instruction for transferring SOL from sender to recipient
const transferInstruction = getTransferSolInstruction({
source: sender,
destination: recipient.address,
amount: transferAmount
});
// Create transaction message
const transactionMessage = pipe(
createTransactionMessage({ version: 0 }),
(tx) => setTransactionMessageFeePayerSigner(sender, tx),
(tx) => setTransactionMessageLifetimeUsingBlockhash(latestBlockhash, tx),
(tx) => appendTransactionMessageInstructions([transferInstruction], tx)
);
const signedTransaction =
await signTransactionMessageWithSigners(transactionMessage);
// Decode the messageBytes
const compiledTransactionMessage =
getCompiledTransactionMessageDecoder().decode(signedTransaction.messageBytes);
console.log(JSON.stringify(compiledTransactionMessage, null, 2));
Console
Click to execute the code.

Alla oleva koodi näyttää edellisten koodiesimerkkien tulosteen. Muoto vaihtelee SDK:iden välillä, mutta huomaa, että jokainen ohje sisältää samat vaaditut tiedot.

{
"version": 0,
"header": {
"numSignerAccounts": 1,
"numReadonlySignerAccounts": 0,
"numReadonlyNonSignerAccounts": 1
},
"staticAccounts": [
"HoCy8p5xxDDYTYWEbQZasEjVNM5rxvidx8AfyqA4ywBa",
"5T388jBjovy7d8mQ3emHxMDTbUF8b7nWvAnSiP3EAdFL",
"11111111111111111111111111111111"
],
"lifetimeToken": "EGCWPUEXhqHJWYBfDirq3mHZb4qDpATmYqBZMBy9TBC1",
"instructions": [
{
"programAddressIndex": 2,
"accountIndices": [0, 1],
"data": {
"0": 2,
"1": 0,
"2": 0,
"3": 0,
"4": 128,
"5": 150,
"6": 152,
"7": 0,
"8": 0,
"9": 0,
"10": 0,
"11": 0
}
}
]
}

Transaktiotietojen hakeminen

Lähetyksen jälkeen hae transaktiotiedot käyttämällä transaktioallekirjoitusta ja getTransaction RPC-metodia.

Voit myös löytää transaktion käyttämällä Solana Exploreria.

Transaction Data
{
"blockTime": 1745196488,
"meta": {
"computeUnitsConsumed": 150,
"err": null,
"fee": 5000,
"innerInstructions": [],
"loadedAddresses": {
"readonly": [],
"writable": []
},
"logMessages": [
"Program 11111111111111111111111111111111 invoke [1]",
"Program 11111111111111111111111111111111 success"
],
"postBalances": [989995000, 10000000, 1],
"postTokenBalances": [],
"preBalances": [1000000000, 0, 1],
"preTokenBalances": [],
"rewards": [],
"status": {
"Ok": null
}
},
"slot": 13049,
"transaction": {
"message": {
"header": {
"numReadonlySignedAccounts": 0,
"numReadonlyUnsignedAccounts": 1,
"numRequiredSignatures": 1
},
"accountKeys": [
"8PLdpLxkuv9Nt8w3XcGXvNa663LXDjSrSNon4EK7QSjQ",
"7GLg7bqgLBv1HVWXKgWAm6YoPf1LoWnyWGABbgk487Ma",
"11111111111111111111111111111111"
],
"recentBlockhash": "7ZCxc2SDhzV2bYgEQqdxTpweYJkpwshVSDtXuY7uPtjf",
"instructions": [
{
"accounts": [0, 1],
"data": "3Bxs4NN8M2Yn4TLb",
"programIdIndex": 2,
"stackHeight": null
}
],
"indexToProgramIds": {}
},
"signatures": [
"3jUKrQp1UGq5ih6FTDUUt2kkqUfoG2o4kY5T1DoVHK2tXXDLdxJSXzuJGY4JPoRivgbi45U2bc7LZfMa6C4R3szX"
]
},
"version": "legacy"
}

Is this page helpful?

Sisällysluettelo

Muokkaa sivua

Hallinnoi

© 2026 Solana Foundation.
Kaikki oikeudet pidätetään.
Yhdistä