토큰 계정 닫기는 무엇을 하나요?
토큰 계정을 닫으면 계정이 삭제되고 토큰 계정의 rent lamport가 목적지 계정으로 반환됩니다.
계정을 닫기 전에 토큰 잔액이 0이어야 합니다. 동결된 계정도 토큰 잔액이 0이면 닫을 수 있습니다. Wrapped SOL 토큰 계정은 예외로, 기본 SOL을 회수하기 위해 토큰 잔액이 있어도 닫을 수 있습니다. 계정 소유자 또는 계정의 닫기 권한자가 닫기 명령에 서명합니다.
토큰 계정을 닫는 방법
토큰 계정 닫기는 Token Program의 CloseAccount 명령을 사용합니다.
CloseAccount 명령은 소스 계정의 lamport를 목적지 계정으로 전송하고, 소스
계정을 초기화하며, 소스 계정을 삭제합니다. Token Extension Program은 특정 확장에
대해 추가적인 닫기 검사를 적용하지만, 기본 닫기 흐름은 동일합니다.
소스 참조
| 항목 | 설명 | Token Program | Token Extension Program |
|---|---|---|---|
Account | 토큰 계정 상태는 계정 닫기 중에 사용되는 잔액, 네이티브 상태, 닫기 권한을 저장합니다. | 소스 | 소스 |
CloseAccount | 토큰 계정을 닫고 토큰 계정의 lamport를 목적지 계정으로 전송하는 명령입니다. | 소스 | 소스 |
process_close_account | 토큰 계정 닫기를 위한 공유 프로세서 로직입니다. | 소스 | 소스 |
Typescript
아래 Kit 예제는 @solana/kit를 사용하는 권장 접근 방식을 보여줍니다.
@solana/web3.js를 사용하는 레거시 예제는 참고용으로 포함되어 있습니다.
Kit
import { generateKeyPairSigner } from "@solana/kit";import { createLocalClient } from "@solana/kit-client-rpc";import {findAssociatedTokenPda,getCreateAssociatedTokenInstructionAsync,tokenProgram,TOKEN_PROGRAM_ADDRESS} from "@solana-program/token";const client = await createLocalClient().use(tokenProgram());const mint = await generateKeyPairSigner();const destination = client.payer.address;const [tokenAccount] = await findAssociatedTokenPda({mint: mint.address,owner: client.payer.address,tokenProgram: TOKEN_PROGRAM_ADDRESS});const result = await client.token.instructions.closeAccount({account: tokenAccount, // Token account to close.destination, // Account receiving the reclaimed SOL.owner: client.payer // Owner approving the account closure.}).sendTransaction();const tokenAccountData =await client.token.accounts.token.fetchMaybe(tokenAccount);console.log("Mint Address:", mint.address);console.log("\nToken Account Address:", tokenAccount);console.log("Token Account:", tokenAccountData);console.log("\nDestination Address:", destination);console.log("\nTransaction Signature:", result.context.signature);
Console
Click to execute the code.
Web3.js
import { Connection, Keypair, LAMPORTS_PER_SOL } from "@solana/web3.js";import {createAssociatedTokenAccount,closeAccount,createMint,TOKEN_PROGRAM_ID} from "@solana/spl-token";const result = await closeAccount(connection, // Connection to the local validator.feePayer, // Account paying transaction fees.associatedTokenAccount, // Token account to close.destination, // Account receiving the reclaimed SOL.feePayer, // Owner approving the account closure.[], // Additional multisig signers.{commitment: "confirmed" // Confirmation options for the transaction.},TOKEN_PROGRAM_ID // Token program to invoke.);const tokenAccountData = await connection.getAccountInfo(associatedTokenAccount,"confirmed");console.log("Mint Address:", mintPubkey.toBase58());console.log("\nAssociated Token Account Address:",associatedTokenAccount.toBase58());console.log("Associated Token Account:", tokenAccountData);console.log("\nDestination Address:", destination.toBase58());console.log("\nTransaction Signature:", result);
Console
Click to execute the code.
Rust
Rust
use anyhow::Result;use solana_client::nonblocking::rpc_client::RpcClient;use solana_commitment_config::CommitmentConfig;use solana_sdk::{program_pack::Pack,signature::{Keypair, Signer},transaction::Transaction,};use solana_system_interface::instruction::create_account;use spl_associated_token_account_interface::{address::get_associated_token_address, instruction::create_associated_token_account,};use spl_token_interface::{id as token_program_id,instruction::{close_account, initialize_mint},state::{Account, Mint},};#[tokio::main]async fn main() -> Result<()> {let client = RpcClient::new_with_commitment(String::from("http://localhost:8899"),CommitmentConfig::confirmed(),);let associated_token_address = get_associated_token_address(&fee_payer.pubkey(), &mint.pubkey());let destination = fee_payer.pubkey();let transaction = Transaction::new_signed_with_payer(&[close_account(&token_program_id(), // Token program to invoke.&associated_token_address, // Token account to close.&destination, // Account receiving the reclaimed SOL.&fee_payer.pubkey(), // Owner approving the account closure.&[], // Additional multisig signers.)?,],Some(&fee_payer.pubkey()),&[&fee_payer],latest_blockhash,);let transaction_signature = client.send_and_confirm_transaction(&transaction).await?;let token_data = match client.get_account(&associated_token_address).await {Ok(account) => Some(Account::unpack(&account.data)?),Err(_) => None,};println!("Mint Address: {}", mint.pubkey());println!("\nAssociated Token Account Address: {}",associated_token_address);println!("Associated Token Account: {:#?}", token_data);println!("\nDestination Address: {}", destination);println!("\nTransaction Signature: {}", transaction_signature);Ok(())}
Console
Click to execute the code.
Python
Python
#!/usr/bin/env python3import asyncioimport jsonfrom solana.rpc.async_api import AsyncClientfrom solders.keypair import Keypairfrom solders.message import Messagefrom solders.system_program import create_account, CreateAccountParamsfrom solders.transaction import Transactionfrom spl.token.async_client import AsyncTokenfrom spl.token.instructions import (close_account,CloseAccountParams,create_associated_token_account,get_associated_token_address,initialize_mint,InitializeMintParams,)from spl.token.constants import MINT_LEN, TOKEN_PROGRAM_IDasync def main():rpc = AsyncClient("http://localhost:8899")async with rpc:close_account_instruction = close_account(CloseAccountParams(program_id=TOKEN_PROGRAM_ID, # Token program to invoke.account=token_account_address, # Token account to close.dest=fee_payer.pubkey(), # Account receiving the reclaimed lamports.owner=fee_payer.pubkey(), # Account allowed to close the token account.))latest_blockhash = await rpc.get_latest_blockhash()transaction = Transaction([fee_payer],Message([close_account_instruction], fee_payer.pubkey()),latest_blockhash.value.blockhash,)result = await rpc.send_transaction(transaction)closed_token_account = (await rpc.get_account_info(token_account_address)).valueprint("Mint Address:", mint.pubkey())print("\nToken Account Address:", token_account_address)print("Token Account:")print(json.dumps(closed_token_account, indent=2))print("\nTransaction Signature:", result.value)if __name__ == "__main__":asyncio.run(main())
Console
Click to execute the code.
Is this page helpful?