前言
本文主要介绍ERC721和ERC721A的特性和方法,ERC721和ERC721A都是以太坊上的非同质化代币(NFT)标准,但它们在实现方式和特性上存在一些区别:
数据结构
- ERC721 :每个
tokenId
都单独记录其所有者信息,逻辑清晰,但存在数据冗余。 - ERC721A :采用惰性初始化机制,对于连续的
tokenId
,只在第一个位置记录所有者信息,后续tokenId
的所有者信息通过向前查找补齐。
Gas消耗
- ERC721 :在批量铸造时,每个
tokenId
都需要单独记录所有者信息,导致Gas消耗较高。 - ERC721A:通过优化数据结构,减少了存储操作,显著降低了批量铸造时的Gas消耗。例如,一次性铸造10个NFT时,ERC721的Gas用量接近30万,而ERC721A仅需约11万。
转让成本
- ERC721 :在转让NFT时,直接更新
tokenId
的所有者信息,Gas消耗相对稳定。 - ERC721A:第一次转让时,需要补充记录所有者信息,Gas消耗较高,但后续转让的Gas成本会降低。
兼容性
- ERC721:是广泛使用的标准,与大多数钱包、市场和DApp兼容。
- ERC721A:虽然引入了新的优化,但仍然与ERC721标准兼容,可以无缝集成到现有的NFT生态系统中。
适用场景
- ERC721:适用于对Gas消耗不敏感的NFT项目,尤其是单个铸造或少量铸造的场景。
- ERC721A:更适合需要大量批量铸造的NFT项目,如大型NFT集合或游戏中的批量生成。
代码对比区别
ERC721
- 合约
ini
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
import "@openzeppelin/contracts/token/ERC721/extensions/ERC721URIStorage.sol";
contract MyNFT is ERC721URIStorage {
constructor() ERC721("MyNFT", "MNFT") {}
// 单个铸造
function mint(address to, uint256 tokenId, string memory tokenURI_) public {
_safeMint(to, tokenId);
_setTokenURI(tokenId, tokenURI_);
}
// 批量铸造
function mintBatch(address to, uint256[] memory tokenIds, string[] memory tokenURIs) public {
require(tokenIds.length == tokenURIs.length, "Array lengths do not match");
for (uint256 i = 0; i < tokenIds.length; i++) {
uint256 tokenId = tokenIds[i];
string memory tokenURI_ = tokenURIs[i];
_safeMint(to, tokenId);
_setTokenURI(tokenId, tokenURI_);
}
}
}
- 测试
bash
调用合约步骤:
# 铸造 调用mintBatch
# 验证 调用toTokenURI
# 转账 调用transferFrom
ERC721A
typescript
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;
import "erc721a/contracts/ERC721A.sol";
import "@openzeppelin/contracts/utils/Strings.sol";
contract MyERC721A is ERC721A {
constructor(string memory name_, string memory symbol_) ERC721A(name_, symbol_) {}
// 批量铸造函数
function mint(address to, uint256 quantity) public {
// 在实际应用中,可能需要添加权限控制,例如只有合约所有者或特定角色可以调用
_mint(to, quantity);
}
// 设置 Token URI
function _startTokenId() internal pure override returns (uint256) {
return 1; // 从 1 开始编号
}
function tokenURI(uint256 tokenId) public view override returns (string memory) {
require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");
// 使用 IPFS 哈希值生成 tokenURI
// 示例 IPFS 哈希值:QmHash...
string memory baseURI = "https://ipfs.io/ipfs/QmHash";
return string(abi.encodePacked(baseURI, Strings.toString(tokenId), ".json"));
}
}
测试
bash
调用合约步骤:
# 铸造 调用mint
# 验证 调用toTokenURI
# 转账 调用transferFrom
截图对比
铸造
ERC721(铸造) | ERC721A(铸造) |
---|---|
![]() |
![]() |
铸造gas
ERC721(铸造gas) | ERC721A(铸造gas) |
---|---|
![]() |
![]() |
转账
ERC721(转账gas) | ERC721A(转账gas) |
---|---|
![]() |
![]() |
转账gas
ERC721(转账gas) | ERC721A(转账gas) |
---|---|
![]() |
![]() |
交易费用对比
交易费用对比 | ||
---|---|---|
批量铸造 | 转账 | |
ERC721 | 220865(gas) | 76143 (gas) |
ERC721A | 60429 (gas) | 84043(gas) |
总结
ERC721A通过优化数据结构和存储方式,在批量铸造时显著降低了Gas消耗,但牺牲了部分转让效率。如果项目需要大量批量铸造,ERC721A是一个更好的选择;如果项目更注重单个铸造和转让效率,可以选择传统的ERC721标准。