以太坊作为全球领先的智能合约平台,其核心功能之一便是支持点对点的价值转移——即交易,无论是发送以太币(ETH),还是与智能合约进行交互,都离不开“交易”这一基本操作,而“代码”则是构建、发送和理解这些交易的基石,本文将深入探讨以太坊交易的原理,并通过代码示例展示如何在实际操作中处理以太坊交易。
以太坊交易的核心原理
以太坊交易本质上是一条被签名后广播到整个网络的数据消息,它包含了执行某种操作所需的全部信息,一个标准的以太坊交易通常包含以下关键字段:
- Nonce (序列号):发送方账户发起的交易序号,用于防止重放攻击并确保交易顺序。
- Gas Price ( gas价格):发送者愿意为每单位gas支付的价格,单位是Gwei (1 ETH = 10^9 Gwei),Gas Price越高,交易被矿工打包的优先级通常越高。
- Gas Limit ( gas限制):发送者愿意为这笔交易支付的最大gas量,它决定了交易可以执行的 computational steps(计算步骤)数量,如果交易执行完毕消耗的gas小于Gas Limit,剩余的gas会退还给发送者;如果消耗的gas达到Gas Limit仍未完成,交易会失败,且已消耗的gas不予退还。
- Recipient (接收者地址):
- 对于普通ETH转账,这是接收方的以太坊地址。
- 对于智能合约交互,这是智能合约的地址。
- 如果此项为空(或特定值,如0x00...0),则表示这是一个“合约创建”交易。
- Value (交易值):发送的ETH数量,单位是wei (1 ETH = 10^18 wei)。
- Data (数据字段):
- 对于普通ETH转账,通常为空或特定数据。
- 对于智能合约交互,这是调用合约函数的ABI编码数据,包含了函数选择器和参数。
- Signature (签名):由发送者使用其私钥对交易数据进行签名,证明交易确实由该发送者发起,并确保交易数据的完整性,签名由
v,r,s三个分量组成。
这些字段共同构成了一个交易,并通过P2P网络广播到以太坊的各个节点,由矿工打包进区块并通过执行EVM(以太坊虚拟机)来改变以太坊的状态。
以太坊交易的代码实现
与以太坊交互,最常用的编程语言是Solidity(用于编写智能合约本身),而用于发送交易、查询状态等操作的则通常是JavaScript/TypeScript(配合Web3.js或ethers.js库)、Python(配合web3.py库)等,下面我们以JavaScript中最流行的ethers.js库为例,展示如何发送一个ETH转账交易和如何调用一个智能合约方法。
准备工作:安装ethers.js
npm install ethers
示例1:发送ETH转账交易
假设我们有一个发送者的私钥(注意:实际应用中应使用安全的方式管理和存储私钥,如硬件钱包或环境变量,此处仅作演示)和一个接收者的地址。
const { ethers } = require("ethers");
// 1. 创建provider(连接到以太坊网络,例如Ropsten测试网或主网)
// 这里以使用Infura的示例为例,你需要替换成自己的Infura项目ID
const provider = new ethers.providers.InfuraProvider("goerli", "YOUR_INFURA_PROJECT_ID");
// 2. 创建wallet(发送者账户)
const privateKey = "YOUR_SENDER_PRIVATE_KEY"; // 替换为发送者的私钥
const wallet = new ethers.Wallet(privateKey, provider);
// 3. 接收者地址
const recipientAddress = "0xRecipientAddress1234567890123456789012345678901234567890"; // 替换为接收者地址
// 4. 获取当前nonce(确保交易顺序正确)
const nonce = await wallet.getTransactionCount();
// 5. 构建交易
const tx = {
to: recipientAddress,
value: ethers.utils.parseEther("0.01"), // 转账0.01 ETH
gasLimit: 21000, // 转账ETH的gasLimit通常是21000
gasPrice: await provider.getGasPrice(), // 获取当前建议的gasPrice
nonce: nonce, // 当前nonce
};
// 6. 发送交易并等待确认
const txResponse = await wallet.sendTransaction(tx);
console.log("交易已发送,哈希:", txResponse.hash);
// 7. 等待交易被矿工打包
const txReceipt = await txResponse.wait();
console.log("交易已确认,区块号:", txReceipt.blockNumber);
console.log("实际消耗的gas:", txReceipt.gasUsed.toString());
示例2:调用智能合约方法
假设我们有一个简单的智能合约,名为MyToken,它有一个transfer方法,用于转移代币。
我们需要智能合约的ABI(Application Binary Interface)和地址。
const { ethers } = require("ethers");
// 1. 创建provider和wallet(同上)
const provider = new ethers.providers.InfuraProvider("goerli", "YOUR_INFURA_PROJECT_ID");
const privateKey = "YOUR_SENDER_PRIVATE_KEY";
const wallet = new ethers.Wallet(privateKey, provider);
// 2. 智能合约ABI(简化版,实际需要完整的ABI)
const tokenAbi = [
"function transfer(address to, uint256 amount) returns (bool)",
"function balanceOf(address account) view returns (uint256)"
];
// 智能合约地址
const tokenAddress = "0xYourContractAddress123456789012345678901234567890123456";
// 3. 创建合约实例
const tokenContract = new ethers.Contract(tokenAddress, tokenAbi, wallet);
// 4. 调用合约的transfer方法(这是一个交易,会修改链上状态)
async function transferTokens() {
const recipient = "0xRecipientAddress1234567890123456789012345678901234567890";
const amount = ethers.utils.parseUnits("100", 18); // 假设代币精度是18位小数
console.log(`正在向 ${recipient} 转账 ${amount.toString()} 个代币...`);
// 发送交易调用transfer函数
const transferTx = await tokenContract.transfer(recipient, amount);
console.log("交易已发送,哈希:", transferTx.hash);
// 等待交易确认
const receipt = await transferTx.wait();
console.log("交易已确认,区块号:", receipt.blockNumber);
console.log("Gas 使用量:", receipt.gasUsed.toString());
// 5. 调用合约的查询方法(view/pure函数,不产生交易)
const recipientBalance = await tokenContract.balanceOf(recipient);
console.log(`${recipient} 的新余额: ${recipientBalance.toString()}`);
}
transferTokens().catch(console.error);
