空投是币圈中一种营销手段,项目方将代币免费发放给特定用户群体。为了拿到空投资格,用户通常需要完成一些简单的任务,如测试产品、分享新闻、介绍朋友等。项目方通过空投可以获得种子用户,而用户可以获得一笔财富,两全其美。
因为每次接收空投的用户很多,项目方不可能一笔一笔的转账。利用只能合约批量发放ERC20代币,可以显著提高空投效率。
Airdrop空投合约逻辑非常简单:利用循环,一笔交易将ERC20代币发送给多个地址。合约中包含两个函数:
getSum()函数:返回uint数组的和。
//数组求和函数
function getSum(uint256[] calldata _arr) public pure returns(uint sum) {
for(uint i = 0; i< _arr.length; i++) {
sum = sum + _arr[i];
}
}
multiTransferToken()函数:发送ERC20代币空投,包含3个参数:_token:代币合约地址(address类型)_addresses:接收空投的用户地址数组(address[]类型)_amount:空投数量数组,对应_addresses里每个地址的数量(uint[]类型)
该函数有两个检查:第一个require检查了_addresses和_amounts两个数组长度是否相等;第二个require检查了空投合约的授权额度大于要空投的代币数量总和。
// @notice 向多个地址转账ERC20代币,使用前需要先授权
// @param _token 转账的ERC20代币地址
// @param _addresses 空投地址数组
// @param _amounts 代币数量数组(每个地址的空投数量)
function multiTransferToken(address _token, address[] calldata _addresses, uint256[] calldata _amounts) external {
require(_addresses.length == _amounts.length, "Lengths of Addresses and Amounts NOT EQUAL");
IERC20 token = IERC20(_token); //声明IERC20合约变量
uint _amountSum = getSum(_amounts); //计算空投代币总量
//检查:授权代币数量 > 空投代币总量
require(token.allowance(msg.sender, address(this)) > _amountSum, "Need Approve ERC20 token");
for(uint256 i = 0; i < _addresses.length; i++) {
token.transferFrom(msg.sender, _addresses[i],_amounts[i]);
}
}
multiTransferEth()函数:发送ETH空投,包含两个参数:_addresses:接收空投的用户地址数组(address[]类型)_amount:空投数量数组,对应_addresses里每个地址的数量(uint[]类型)
//向多个地址转账eth
function multiTransferETH(address payable[] calldata _addresses, uint256[] calldata _amounts) public payable {
require(_addresses.length == _amounts.length, "Lengths of Addresses and Amounts NOT EQUAL");
uint _amountSum = getSum(_amounts);
//检查转入ETH等于空投总量
require(msg.value == _amountSum, "Transfer amount error");
//for循环,利用transfer函数发送ETH
for(uint256 i = 0; i < _addresses.length; i++){
// 注释代码有Dos攻击风险, 并且transfer 也是不推荐写法
// Dos攻击 具体参考 https://github.com/AmazingAng/WTF-Solidity/blob/main/S09_DoS/readme.md
// _addresses[i].transfer(_amounts[i]);
(bool success, ) = _addresses[i].call{value: _amounts[i]}("");
if(!success) {
failTransferList[_addresses[i]] = _amounts[i];
}
}
}
1. 部署ERC20代币合约,并给自己mint10000单位代币


2. 部署Airdrop空投合约。

3. 利用ERC20代币合约中的approve()函数,给Airdrop空投合约授权10000单位代币。

4. 执行Airdrop合约的multiTransferToken()函数进行空投,_token填ERC20代币地址,_addresses和_amounts按照以下填写
// _addresses填写
["0xAb8483F64d9C6d1EcF9b849Ae677dD3315835cb2","0x4B20993Bc481177ec7E8f571ceCaE8A9e22C02db"]
// _amounts填写
[100, 200]

5. 利用ERC20合约的balanceOf()函数查询上面用户地址的代币余额,成功变为100和200,空投成功!



