ERC721解读

2024-02-22 22:36
文章标签 解读 erc721

本文主要是介绍ERC721解读,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

NFT(非同质化代币):类似于明朝、宋朝的青花瓷。虽然都是青花瓷。但是都具有唯一的典藏价值。而且价值可能不同。 NFT就是具有唯一价值的代币。

ERC721: 是以太坊规定实现NFT的一种标准了。实现ERC21标准的智能合约就是NFT代币了。

1.接口

1.ERC721

  定义接口参考:ERC 721 - OpenZeppelin 文档

下面是以太坊官方定义的标准,由于就是我写的代码运行环境不支持payable关键字,因此我打算围绕官方接口定义,按照自己要求稍微增删一下。

pragma solidity ^0.4.25;interface ERC721 {///Event///event Transfer(address indexed _from, address indexed _to, uint256 indexed _tokenId);event Approval(address indexed _owner, address indexed _approved, uint256 indexed _tokenId);event ApprovalForAll(address indexed _owner, address indexed _operator, bool _approved);///Function///function balanceOf(address _owner) external view returns (uint256); // 返回所有者代币的总个数function ownerOf(uint256 _tokenId) external view returns (address); // 返回代币id对应所有者的账户地址// 安全的转账//  _to:是已经被指定 id 代币的所有者授予的账户 and (接受者不是智能合约 or 接受者实现ERC721Receiver接口的智能合约// 将给定id的代币转移到接受者账户// data是元数据,可有可不有(我觉得)function safeTransferFrom(address _from, address _to, uint256 _tokenId, bytes data) external; function safeTransferFrom(address _from, address _to, uint256 _tokenId) external;// 这个转账对比上述安全转账(少了一个接受者地址实现是否是ERC721Receiver接口的智能合约地址的判断)function transferFrom(address _from, address _to, uint256 _tokenId) external;// 授权将代币转移到另一个账户的权限function approve(address _approved, uint256 _tokenId) external;// 授权接受者使用所有代币function setApprovalForAll(address _operator, bool _approved) external;// 返回授权指定id 代币的接受者账户function getApproved(uint256 _tokenId) external view returns (address);// 判断某账户代币的拥有者是否能被某账户全部使用function isApprovedForAll(address _owner, address _operator) external view returns (bool);
}

2.ERC721Metadata

以下就是ERC21的元数据接口,这是可选地。名称、标识符、每一个token对应的tokenURI。

pragma solidity ^0.4.25;interface ERC721Metadata {function name() external view returns (string);function symbol() external view returns (string);function tokenURI(uint256 tokenId) external view returns (string); // 返回指定id的代币所对对应的uri
}

3.ERC721Enumerable

另一个额外的可选接口是枚举, 它包含了按索引获取到对应的代币。

pragma solidity ^0.4.25;interface ERC721Enumerable {// 确定合约当前全部的nft数量(出去销毁)function totalSupply() external view returns (uint256);// 从代币列表返回第n个代币function tokenByIndex(uint256 _index) external view returns (uint256);// 返回所有者代币列表的第n个代币function tokenOfOwnerByIndex(address _owner, uint256 _index) external view returns (uint256);
}

4.ERC721Receiver

pragma solidity ^0.4.25;// 资产合约
interface ERC721Receiver {function onERC721Received(address operator, address from, uint256 tokenId, bytes data) externalreturns (bytes);
}

2.实现

1.Jzm721

这是我针对ERC721接口的合约实现。基本满足官方接口标准。

pragma solidity ^0.4.25;
import "./ERC721.sol";
import "./ERC721Metadata.sol";
contract Jzm721 is ERC721,ERC721Metadata {///Filed///string public name;string public symbol;uint256  nftCount;mapping (address => uint[])  balanceMap; // owner => tokenId[]mapping (uint256=>string)  tokenURIMap; // tokenId => tokenURImapping (uint256=>address)  tokenIdMap; // tokenId => ownermapping (uint256 => address) approveMap; // tokenId => operator(经营方)mapping (address=>mapping (address=>bool)) approveAllMap; // operator =>(owner => true/false)///Function///constructor(string memory _name,string memory _symbol) public {name = _name;symbol = _symbol;}function name() external view returns (string) {return name;}function symbol() external view returns (string) {return symbol;}function tokenURI(uint256 tokenId) external view returns (string) {return tokenURIMap[tokenId];}// 创建代币function mint(address _owner,string _tokenURI) external  returns (uint256) {require(_owner != address(0),"owner is not empty address!");uint256 tokenId = _mint(_owner);_setTokenURI(tokenId, _tokenURI);return tokenId;}function balanceOf(address _owner) external view returns (uint256) {return balanceMap[_owner].length;}function ownerOf(uint256 _tokenId) external view returns (address) {return tokenIdMap[_tokenId];}// 这里我忽略了data这个元数据的作用function safeTransferFrom(address _from, address _to, uint256 _tokenId, bytes data) external {_transferFrom(_from, _to, _tokenId);if(_isContractAdd(_to)) {if(_checkIfFunctionExists(_to)) {_externalTransfer(_from, _to, _tokenId);emit Transfer(_from, _to, _tokenId);}  }}function safeTransferFrom(address _from, address _to, uint256 _tokenId) external {_transferFrom(_from, _to, _tokenId);if(_isContractAdd(_to)) {if(_checkIfFunctionExists(_to)) {_externalTransfer(_from, _to, _tokenId);emit Transfer(_from, _to, _tokenId);}}}function transferFrom(address _from, address _to, uint256 _tokenId) external {_transferFrom(_from, _to, _tokenId);}function approve(address _approved, uint256 _tokenId) external {require(_approved != address(0),"approved is not empty address!");address owner = msg.sender;approveMap[_tokenId] = _approved; emit Approval(owner,_approved,_tokenId);}function setApprovalForAll(address _operator, bool _approved) external {require(_operator != address(0),"operator is not empty address!");address owner = msg.sender;approveAllMap[_operator][owner] = _approved;emit  ApprovalForAll(owner,_operator,_approved);}function getApproved(uint256 _tokenId) external view returns (address) {return _getApproved(_tokenId);}function isApprovedForAll(address _owner, address _operator) external view returns (bool) {return _isApprovedForAll(_owner,_operator);}function _checkIfFunctionExists(address _add)public  returns (bool) {bytes4 functionSelector = bytes4(keccak256("onERC721Received(address,address,uint256,bytes)")); // 函数选择器,基础原型前4个字节bytes memory data = abi.encodeWithSelector(functionSelector, address(0),address(0),0,"");bool success = _add.call(data);return success;}function _transferFrom(address _from, address _to, uint256 _tokenId) private {// from不能是零地址。require(_from != address(0),"from is not empty address!");// to不能是零地址。require(_to != address(0),"from is not empty address!");// tokenId令牌必须存在并由from拥有。require(tokenIdMap[_tokenId] == _from,"The tokenId  must exist and be owned by from!");// 接受者一方不是合约地址if (!_isContractAdd(_to)) {if (_isApproved(_from, _to, _tokenId)) { // TODO_externalTransfer(_from, _to, _tokenId);emit Transfer(_from, _to, _tokenId);}}}function _externalTransfer(address _from,address _to,uint256 _tokenId) private {// 删除代币批准if(_getApproved(_tokenId) == _to) {approveMap[_tokenId] = address(0);   }// 转账_deleteAccountToken(_from,_tokenId);tokenIdMap[_tokenId] = _to;balanceMap[_to].push(_tokenId);}function _deleteAccountToken(address _owner,uint256 _tokenId) private {uint256[] storage tokenIds = balanceMap[_owner];uint len =  tokenIds.length;for (uint i = 0; i < len; i++) {if(tokenIds[i] == _tokenId) {// 交换uint swap;swap = tokenIds[i];tokenIds[i] = tokenIds[len - 1];tokenIds[len - 1] = swap;}}tokenIds.length--;}function _mint(address _owner) private  returns (uint256) {nftCount += 1;uint256 tokenId = nftCount + block.timestamp;balanceMap[_owner].push(tokenId);tokenIdMap[tokenId] = _owner;emit Transfer(address(0),_owner,tokenId);return tokenId;}function _setTokenURI(uint256 _tokenId,string _tokenURI) private{tokenURIMap[_tokenId] = _tokenURI;}// 判断该地址是否合约地址function _isContractAdd(address  _addr) private view returns (bool) {uint size;assembly {size := extcodesize(_addr) // 返回地址关联代码的长度}return size > 0;}function _getApproved(uint256 _tokenId) private view returns(address) {return approveMap[_tokenId];}function _isApproved(address _owner, address _operator,uint256 _tokenId) private view returns (bool) {bool approved = _getApproved(_tokenId) == _operator;return approved || _isApprovedForAll(_owner, _operator);}function _isApprovedForAll(address _owner, address _operator) private view returns(bool) {return approveAllMap[_operator][_owner];}}

2.Jzm721Receiver

在这里我就是想要满足合约地址的合约实现ERC721Receiver接口的标准。原合约,这里该函数涉及代币的转账,由于环境的原因,不支持payable关键字,我这里是无法满足的。

pragma solidity ^0.4.25;
import "./ERC721Receiver.sol";contract Jzm721Receiver is ERC721Receiver {function onERC721Received(address operator, address from, uint256 tokenId, bytes data) externalreturns (bytes) {return data;}
}

这篇关于ERC721解读的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/736705

相关文章

MCU7.keil中build产生的hex文件解读

1.hex文件大致解读 闲来无事,查看了MCU6.用keil新建项目的hex文件 用FlexHex打开 给我的第一印象是:经过软件的解释之后,发现这些数据排列地十分整齐 :02000F0080FE71:03000000020003F8:0C000300787FE4F6D8FD75810702000F3D:00000001FF 把解释后的数据当作十六进制来观察 1.每一行数据

Java ArrayList扩容机制 (源码解读)

结论:初始长度为10,若所需长度小于1.5倍原长度,则按照1.5倍扩容。若不够用则按照所需长度扩容。 一. 明确类内部重要变量含义         1:数组默认长度         2:这是一个共享的空数组实例,用于明确创建长度为0时的ArrayList ,比如通过 new ArrayList<>(0),ArrayList 内部的数组 elementData 会指向这个 EMPTY_EL

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

GPT系列之:GPT-1,GPT-2,GPT-3详细解读

一、GPT1 论文:Improving Language Understanding by Generative Pre-Training 链接:https://cdn.openai.com/research-covers/languageunsupervised/language_understanding_paper.pdf 启发点:生成loss和微调loss同时作用,让下游任务来适应预训

LLM系列 | 38:解读阿里开源语音多模态模型Qwen2-Audio

引言 模型概述 模型架构 训练方法 性能评估 实战演示 总结 引言 金山挂月窥禅径,沙鸟听经恋法门。 小伙伴们好,我是微信公众号《小窗幽记机器学习》的小编:卖铁观音的小男孩,今天这篇小作文主要是介绍阿里巴巴的语音多模态大模型Qwen2-Audio。近日,阿里巴巴Qwen团队发布了最新的大规模音频-语言模型Qwen2-Audio及其技术报告。该模型在音频理解和多模态交互

文章解读与仿真程序复现思路——电力自动化设备EI\CSCD\北大核心《考虑燃料电池和电解槽虚拟惯量支撑的电力系统优化调度方法》

本专栏栏目提供文章与程序复现思路,具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源程序擅长文章解读,论文与完整源程序,等方面的知识,电网论文源程序关注python

速通GPT-3:Language Models are Few-Shot Learners全文解读

文章目录 论文实验总览1. 任务设置与测试策略2. 任务类别3. 关键实验结果4. 数据污染与实验局限性5. 总结与贡献 Abstract1. 概括2. 具体分析3. 摘要全文翻译4. 为什么不需要梯度更新或微调⭐ Introduction1. 概括2. 具体分析3. 进一步分析 Approach1. 概括2. 具体分析3. 进一步分析 Results1. 概括2. 具体分析2.1 语言模型

Open-Sora代码详细解读(1):解读DiT结构

Diffusion Models专栏文章汇总:入门与实战 前言:目前开源的DiT视频生成模型不是很多,Open-Sora是开发者生态最好的一个,涵盖了DiT、时空DiT、3D VAE、Rectified Flow、因果卷积等Diffusion视频生成的经典知识点。本篇博客从Open-Sora的代码出发,深入解读背后的原理。 目录 DiT相比于Unet的关键改进点 Token化方

Transformer从零详细解读

Transformer从零详细解读 一、从全局角度概况Transformer ​ 我们把TRM想象为一个黑盒,我们的任务是一个翻译任务,那么我们的输入是中文的“我爱你”,输入经过TRM得到的结果为英文的“I LOVE YOU” ​ 接下来我们对TRM进行细化,我们将TRM分为两个部分,分别为Encoders(编码器)和Decoders(解码器) ​ 在此基础上我们再进一步细化TRM的

过滤器:活性碳过滤器的技术参数详细解读

活性碳过滤器是一种罐体的机械过滤器,外壳一般为不锈钢或者玻璃钢,内部填充活性炭,用来过滤水中的游离物、微生物、部分重金属离子,并能有效降低水的色度。活性炭过滤器是一种较常用的水处理设备,作为水处理脱盐系统前处理能够吸附前级过滤中无法去除的余氯,可有效保证后级设备使用寿命,提高出水水质,防止污染,特别是防止后级反渗透膜,离子交换树脂等的游离态余氧中毒污染。同时还吸附从前级泄漏过来的小分子有机物等