ERC20代币 #
一、ERC20标准概述 #
1.1 标准接口 #
solidity
interface IERC20 {
function totalSupply() external view returns (uint256);
function balanceOf(address account) external view returns (uint256);
function transfer(address recipient, uint256 amount) external returns (bool);
function allowance(address owner, address spender) external view returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
}
1.2 核心功能 #
| 功能 | 说明 |
|---|---|
| totalSupply | 代币总供应量 |
| balanceOf | 查询余额 |
| transfer | 转账 |
| approve | 授权额度 |
| allowance | 查询授权额度 |
| transferFrom | 授权转账 |
二、完整实现 #
2.1 基础ERC20 #
solidity
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
contract ERC20 {
string public name;
string public symbol;
uint8 public decimals;
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
mapping(address => mapping(address => uint256)) public allowance;
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
constructor(
string memory _name,
string memory _symbol,
uint8 _decimals,
uint256 _totalSupply
) {
name = _name;
symbol = _symbol;
decimals = _decimals;
totalSupply = _totalSupply * 10 ** uint256(_decimals);
balanceOf[msg.sender] = totalSupply;
emit Transfer(address(0), msg.sender, totalSupply);
}
function transfer(address to, uint256 amount) public returns (bool) {
require(to != address(0), "ERC20: transfer to zero address");
require(balanceOf[msg.sender] >= amount, "ERC20: insufficient balance");
balanceOf[msg.sender] -= amount;
balanceOf[to] += amount;
emit Transfer(msg.sender, to, amount);
return true;
}
function approve(address spender, uint256 amount) public returns (bool) {
require(spender != address(0), "ERC20: approve to zero address");
allowance[msg.sender][spender] = amount;
emit Approval(msg.sender, spender, amount);
return true;
}
function transferFrom(
address from,
address to,
uint256 amount
) public returns (bool) {
require(from != address(0), "ERC20: transfer from zero address");
require(to != address(0), "ERC20: transfer to zero address");
require(balanceOf[from] >= amount, "ERC20: insufficient balance");
require(allowance[from][msg.sender] >= amount, "ERC20: insufficient allowance");
balanceOf[from] -= amount;
balanceOf[to] += amount;
allowance[from][msg.sender] -= amount;
emit Transfer(from, to, amount);
return true;
}
}
2.2 使用OpenZeppelin #
solidity
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
contract MyToken is ERC20, Ownable {
uint8 private _decimals;
constructor(
string memory name,
string memory symbol,
uint8 decimals_,
uint256 initialSupply
) ERC20(name, symbol) {
_decimals = decimals_;
_mint(msg.sender, initialSupply * 10 ** decimals_);
}
function decimals() public view virtual override returns (uint8) {
return _decimals;
}
function mint(address to, uint256 amount) public onlyOwner {
_mint(to, amount);
}
function burn(uint256 amount) public {
_burn(msg.sender, amount);
}
}
三、扩展功能 #
3.1 可铸造/销毁 #
solidity
import "@openzeppelin/contracts/token/ERC20/extensions/ERC20Burnable.sol";
import "@openzeppelin/contracts/token/ERC20/extensions/ERC20Capped.sol";
contract MintableToken is ERC20, ERC20Burnable, ERC20Capped, Ownable {
constructor(
string memory name,
string memory symbol,
uint256 cap
) ERC20(name, symbol) ERC20Capped(cap) {}
function mint(address to, uint256 amount) public onlyOwner {
_mint(to, amount);
}
}
3.2 Pausable #
solidity
import "@openzeppelin/contracts/token/ERC20/extensions/ERC20Pausable.sol";
contract PausableToken is ERC20, ERC20Pausable, Ownable {
function pause() public onlyOwner {
_pause();
}
function unpause() public onlyOwner {
_unpause();
}
}
四、最佳实践 #
4.1 安全考虑 #
| 考虑点 | 说明 |
|---|---|
| 重入保护 | 使用ReentrancyGuard |
| 溢出检查 | 使用Solidity 0.8.x |
| 权限控制 | 使用Ownable或AccessControl |
| 暂停功能 | 添加紧急停止 |
4.2 Gas优化 #
solidity
// 使用uint256而非uint8节省Gas
uint256 private _totalSupply;
// 批量转账
function batchTransfer(
address[] memory recipients,
uint256[] memory amounts
) public {
require(recipients.length == amounts.length, "Length mismatch");
for (uint256 i = 0; i < recipients.length; i++) {
transfer(recipients[i], amounts[i]);
}
}
五、总结 #
ERC20要点:
| 功能 | 说明 |
|---|---|
| 核心接口 | 6个必需函数 |
| 事件 | Transfer和Approval |
| 扩展 | 铸造、销毁、暂停等 |
| 安全 | 重入保护、权限控制 |
接下来,让我们学习ERC721 NFT!
最后更新:2026-03-27