白天搞智能合约,晚上撸前端代码,只要咖啡还续着,凌晨照样在线!会说 Solidity、PHP、Node.js,还有 Vue 和 HTML 的“方言”,代码不怕我不写,就怕写完跑太快。Bug?那都是小场面,一出手就搞定。我的宗旨是——交付准时,调试无敌,客户满意才是真理!
15. 批量转账
这一讲,我们将介绍用ethers.js进行批量转账。
Airdrop合约
这里简单介绍下Airdrop合约,细节可以去Solidity教程中看。我们会用到2个函数:
multiTransferETH():批量发送ETH,包含2个参数:_addresses:接收空投的用户地址数组(address[]类型)_amounts:空投数量数组,对应_addresses里每个地址的数量(uint[]类型)
multiTransferToken()函数:批量发送ERC20代币,包含3个参数:_token:代币合约地址(address类型)_addresses:接收空投的用户地址数组(address[]类型)_amounts:空投数量数组,对应_addresses里每个地址的数量(uint[]类型)
我们在Goerli测试网部署了一个Airdrop合约,地址为:
0x71C2aD976210264ff0468d43b198FD69772A25fa
批量转账
下面我们写一个脚本,调用Airdrop合约将ETH(原生代币)和WETH(ERC20代币)转账给20个地址。
创建HD钱包,用于批量生成地址。
console.log("\n1. 创建HD钱包") // 通过助记词生成HD钱包 const mnemonic = `air organ twist rule prison symptom jazz cheap rather dizzy verb glare jeans orbit weapon universe require tired sing casino business anxiety seminar hunt` const hdNode = ethers.HDNodeWallet.fromPhrase(mnemonic) console.log(hdNode);
利用HD钱包,生成20个钱包地址。
console.log("\n2. 通过HD钱包派生20个钱包") const numWallet = 20 // 派生路径:m / purpose' / coin_type' / account' / change / address_index // 我们只需要切换最后一位address_index,就可以从hdNode派生出新钱包 let basePath = "m/44'/60'/0'/0"; let addresses = []; for (let i = 0; i < numWallet; i++) { let hdNodeNew = hdNode.derivePath(basePath + "/" + i); let walletNew = new ethers.Wallet(hdNodeNew.privateKey); addresses.push(walletNew.address); } console.log(addresses) const amounts = Array(20).fill(ethers.parseEther("0.0001")) console.log(`发送数额:${amounts}`)
创建provider和wallet,发送代币用。
const ALCHEMY_GOERLI_URL = 'https://eth-goerli.alchemyapi.io/v2/GlaeWuylnNM3uuOo-SAwJxuwTdqHaY5l'; const provider = new ethers.JsonRpcProvider(ALCHEMY_GOERLI_URL); // 利用私钥和provider创建wallet对象 // 如果这个钱包没goerli测试网ETH了 // 请使用自己的小号钱包测试,钱包地址: 0x338f8891D6BdC58eEB4754352459cC461EfD2a5E ,请不要给此地址发送任何ETH // 注意不要把自己的私钥上传到github上 const privateKey = '0x21ac72b6ce19661adf31ef0d2bf8c3fcad003deee3dc1a1a64f5fa3d6b049c06' const wallet = new ethers.Wallet(privateKey, provider)创建Airdrop合约。
// Airdrop的ABI const abiAirdrop = [ "function multiTransferToken(address,address[],uint256[]) external", "function multiTransferETH(address[],uint256[]) public payable", ]; // Airdrop合约地址(Goerli测试网) const addressAirdrop = '0x71C2aD976210264ff0468d43b198FD69772A25fa' // Airdrop Contract // 声明Airdrop合约 const contractAirdrop = new ethers.Contract(addressAirdrop, abiAirdrop, wallet)创建WETH合约。
// WETH的ABI const abiWETH = [ "function balanceOf(address) public view returns(uint)", "function transfer(address, uint) public returns (bool)", "function approve(address, uint256) public returns (bool)" ]; // WETH合约地址(Goerli测试网) const addressWETH = '0xB4FBF271143F4FBf7B91A5ded31805e42b2208d6' // WETH Contract // 声明WETH合约 const contractWETH = new ethers.Contract(addressWETH, abiWETH, wallet)读取一个地址的ETH和WETH余额。
console.log("\n3. 读取一个地址的ETH和WETH余额") //读取WETH余额 const balanceWETH = await contractWETH.balanceOf(addresses[10]) console.log(`WETH持仓: ${ethers.formatEther(balanceWETH)}\n`) //读取ETH余额 const balanceETH = await provider.getBalance(addresses[10]) console.log(`ETH持仓: ${ethers.formatEther(balanceETH)}\n`)
调用
multiTransferETH()函数,给每个钱包转0.0001 ETH,可以看到发送后余额发生变化。console.log("\n4. 调用multiTransferETH()函数,给每个钱包转 0.0001 ETH") // 发起交易 const tx = await contractAirdrop.multiTransferETH(addresses, amounts, {value: ethers.parseEther("0.002")}) // 等待交易上链 await tx.wait() // console.log(`交易详情:`) // console.log(tx) const balanceETH2 = await provider.getBalance(addresses[10]) console.log(`发送后该钱包ETH持仓: ${ethers.formatEther(balanceETH2)}\n`)
调用multiTransferToken()函数,给每个钱包转
0.0001 WETH,可以看到发送后余额发生变化。console.log("\n5. 调用multiTransferToken()函数,给每个钱包转 0.0001 WETH") // 先approve WETH给Airdrop合约 const txApprove = await contractWETH.approve(addressAirdrop, ethers.parseEther("1")) await txApprove.wait() // 发起交易 const tx2 = await contractAirdrop.multiTransferToken(addressWETH, addresses, amounts) // 等待交易上链 await tx2.wait() // console.log(`交易详情:`) // console.log(tx2) // 读取WETH余额 const balanceWETH2 = await contractWETH.balanceOf(addresses[10]) console.log(`发送后该钱包WETH持仓: ${ethers.formatEther(balanceWETH2)}\n`)
完整代码
import { ethers } from "ethers";
// 1. 创建HD钱包
console.log("\n1. 创建HD钱包")
// 通过助记词生成HD钱包
const mnemonic = `air organ twist rule prison symptom jazz cheap rather dizzy verb glare jeans orbit weapon universe require tired sing casino business anxiety seminar hunt`
const hdNode = ethers.HDNodeWallet.fromPhrase(mnemonic)
console.log(hdNode);
// 2. 获得20个钱包的地址
console.log("\n2. 通过HD钱包派生20个钱包")
const numWallet = 20
// 派生路径:m / purpose' / coin_type' / account' / change / address_index
// 我们只需要切换最后一位address_index,就可以从hdNode派生出新钱包
let basePath = "m/44'/60'/0'/0";
let addresses = [];
for (let i = 0; i < numWallet; i++) {
let hdNodeNew = hdNode.derivePath(basePath + "/" + i);
let walletNew = new ethers.Wallet(hdNodeNew.privateKey);
addresses.push(walletNew.address);
}
console.log(addresses)
const amounts = Array(20).fill(ethers.parseEther("0.0001"))
console.log(`发送数额:${amounts}`)
// 3. 创建provider和wallet,发送代币用
const ALCHEMY_GOERLI_URL = 'https://eth-goerli.alchemyapi.io/v2/GlaeWuylnNM3uuOo-SAwJxuwTdqHaY5l';
const provider = new ethers.JsonRpcProvider(ALCHEMY_GOERLI_URL);
// 利用私钥和provider创建wallet对象
// 如果这个钱包没goerli测试网ETH了
// 请使用自己的小号钱包测试,钱包地址: 0x338f8891D6BdC58eEB4754352459cC461EfD2a5E ,请不要给此地址发送任何ETH
// 注意不要把自己的私钥上传到github上
const privateKey = '0x227dbb8586117d55284e26620bc76534dfbd2394be34cf4a09cb775d593b6f2b'
const wallet = new ethers.Wallet(privateKey, provider)
// 4. 声明Airdrop合约
// Airdrop的ABI
const abiAirdrop = [
"function multiTransferToken(address,address[],uint256[]) external",
"function multiTransferETH(address[],uint256[]) public payable",
];
// Airdrop合约地址(Goerli测试网)
const addressAirdrop = '0x71C2aD976210264ff0468d43b198FD69772A25fa' // Airdrop Contract
// 声明Airdrop合约
const contractAirdrop = new ethers.Contract(addressAirdrop, abiAirdrop, wallet)
// 5. 声明WETH合约
// WETH的ABI
const abiWETH = [
"function balanceOf(address) public view returns(uint)",
"function transfer(address, uint) public returns (bool)",
"function approve(address, uint256) public returns (bool)"
];
// WETH合约地址(Goerli测试网)
const addressWETH = '0xB4FBF271143F4FBf7B91A5ded31805e42b2208d6' // WETH Contract
// 声明WETH合约
const contractWETH = new ethers.Contract(addressWETH, abiWETH, wallet)
const main = async () => {
// 6. 读取一个地址的ETH和WETH余额
console.log("\n3. 读取一个地址的ETH和WETH余额")
//读取WETH余额
const balanceWETH = await contractWETH.balanceOf(addresses[10])
console.log(`WETH持仓: ${ethers.formatEther(balanceWETH)}\n`)
//读取ETH余额
const balanceETH = await provider.getBalance(addresses[10])
console.log(`ETH持仓: ${ethers.formatEther(balanceETH)}\n`)
const myETH = await provider.getBalance(wallet)
const myToken = await contractWETH.balanceOf(wallet.getAddress())
// 如果钱包ETH足够和WETH足够
if(ethers.formatEther(myETH) > 0.002 && ethers.formatEther(myToken) >= 0.002){
// 7. 调用multiTransferETH()函数,给每个钱包转 0.0001 ETH
console.log("\n4. 调用multiTransferETH()函数,给每个钱包转 0.0001 ETH")
// 发起交易
const tx = await contractAirdrop.multiTransferETH(addresses, amounts, {value: ethers.parseEther("0.002")})
// 等待交易上链
await tx.wait()
// console.log(`交易详情:`)
// console.log(tx)
const balanceETH2 = await provider.getBalance(addresses[10])
console.log(`发送后该钱包ETH持仓: ${ethers.formatEther(balanceETH2)}\n`)
// 8. 调用multiTransferToken()函数,给每个钱包转 0.0001 WETH
console.log("\n5. 调用multiTransferToken()函数,给每个钱包转 0.0001 WETH")
// 先approve WETH给Airdrop合约
const txApprove = await contractWETH.approve(addressAirdrop, ethers.parseEther("1"))
await txApprove.wait()
// 发起交易
const tx2 = await contractAirdrop.multiTransferToken(addressWETH, addresses, amounts)
// 等待交易上链
await tx2.wait()
// console.log(`交易详情:`)
// console.log(tx2)
// 读取WETH余额
const balanceWETH2 = await contractWETH.balanceOf(addresses[10])
console.log(`发送后该钱包WETH持仓: ${ethers.formatEther(balanceWETH2)}\n`)
}else{
// 如果ETH和WETH不足
console.log("ETH不足,请使用自己的小号钱包测试,并兑换一些WETH")
console.log("1. chainlink水龙头: https://faucets.chain.link/goerli")
console.log("2. paradigm水龙头: https://faucet.paradigm.xyz/")
}
}
main()
