ERC721解读

news/发布时间2024/5/16 8:04:54

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;}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.bcls.cn/QvyD/5556.shtml

如若内容造成侵权/违法违规/事实不符,请联系编程老四网进行投诉反馈email:xxxxxxxx@qq.com,一经查实,立即删除!

相关文章

sentinel中监听器的运用--规则管理

sentinel中监听器的运用–规则管理 规则结构 类图关系 类关系图如下 Rule 将规则抽象成一个类, 规则与资源是紧密关联的, 也就是说规则作用于资源。因此, 我们需要将规则表示为一个类, 并包含一个获取资源的方法 这里采用接口的原因就是规则是一个抽象概念而非具体实现。…

【C进阶】顺序表详解

文章目录 &#x1f4dd;线性表的概念&#x1f320; 顺序表&#x1f309;顺序表的概念 &#x1f320;声明--接口&#x1f309;启动&#x1f320;初始化&#x1f309;扩容&#x1f320;尾插&#x1f309; 打印&#x1f320;销毁&#x1f309; 尾删&#x1f320;头插&#x1f309;…

kitti数据可视化

数据下载 The KITTI Vision Benchmark Suite 这里以 2011_09_26_drive_0005 (0.6 GB)数据为参考&#xff0c;下载[syncedrectified data] [calibration] 数据。 下载完毕之后解压&#xff0c;然后将calibration文件解压后的结果放在如下目录下&#xff0c; 下载kitti2bag包 …

《Docker 简易速速上手小册》第4章 Docker 容器管理(2024 最新版)

文章目录 4.1 容器生命周期管理4.1.1 重点基础知识4.1.2 重点案例&#xff1a;启动并管理 Python Flask 应用容器4.1.3 拓展案例 1&#xff1a;调试运行中的容器4.1.4 拓展案例 2&#xff1a;优雅地停止和清理容器 4.2 容器数据管理与持久化4.2.1 重点基础知识4.2.2 重点案例&a…

数字孪生的技术开发平台

数字孪生的开发平台可以基于各种软件和硬件工具来实现&#xff0c;这些平台提供了丰富的功能和工具&#xff0c;帮助开发人员构建、部署和管理数字孪生系统&#xff0c;根据具体的需求和技术要求&#xff0c;开发人员可以选择合适的平台进行开发工作。以下列举了一些常见的数字…

离散数学——树思维导图

离散数学——树思维导图 文章目录 前言内容大纲参考 前言 这是当初学习离散数学时整理的笔记大纲&#xff0c;其中包含了自己对于一些知识点的体悟。现将其放在这里作为备份&#xff0c;也希望能够对你有所帮助。 当初记录这些笔记只是为了在复习时更快地找到对应的知识点。…

蓝桥杯:真题讲解2(C++版)附带解析

星系炸弹 来自&#xff1a;2015年六届省赛大学B组真题&#xff08;共6道题) 分析&#xff1a;这题涉及到平年和闰年的知识&#xff0c;如果我们要解这题&#xff0c;首先要知道每月有多少天&#xff0c;其实也就是看2月份的天数&#xff0c;其它月份的天数都是一样的&#xff…

铭瑄科技——为星闪技术发展与应用带来新推力

随着智能化生活逐渐普及&#xff0c;无线通信不仅是不仅是信息时代的重要基础设施&#xff0c;而且是推动社会向智能化发展的核心力量之一&#xff0c;其中短距无线通信更是推动未来智能化发展的关键。 为积极推动未来硬件智能化、产业智能化发展&#xff0c;铭瑄正式宣布成为星…

微信小程序 --- 自定义组件

自定义组件 1. 创建-注册-使用组件 组件介绍 小程序目前已经支持组件化开发&#xff0c;可以将页面中的功能模块抽取成自定义组件&#xff0c;以便在不同的页面中重复使用&#xff1b; 也可以将复杂的页面拆分成多个低耦合的模块&#xff0c;有助于代码维护。 开发中常见的…

vue3 使用qrcodejs2-fix生成二维码并可下载保存

直接上代码 <el-button click‘setEwm’>打开弹框二维码</el-button><el-dialog v-model"centerDialogVisible" align-center ><div class"code"><div class"content" id"qrCodeUrl" ref"qrCodeUrl&q…

【MySQL面试复习】谈一谈你对SQL的优化经验

系列文章目录 在MySQL中&#xff0c;如何定位慢查询&#xff1f; 发现了某个SQL语句执行很慢&#xff0c;如何进行分析&#xff1f; 了解过索引吗&#xff1f;(索引的底层原理)/B 树和B树的区别是什么&#xff1f; 什么是聚簇索引&#xff08;聚集索引&#xff09;和非聚簇索引…

免费享受企业级安全:雷池社区版WAF,高效专业的Web安全的方案

网站安全成为了每个企业及个人不可忽视的重要议题。 随着网络攻击手段日益狡猾和复杂&#xff0c;选择一个强大的安全防护平台变得尤为关键。 推荐的雷池社区版——一个为网站提供全面安全防护解决方案的平台&#xff0c;它不仅具备高效的安全防护能力&#xff0c;还让网站安…

2024.2.29 模拟实现 RabbitMQ —— 项目展示

目录 项目介绍 核心功能 核心技术 演示直接交换机 演示扇出交换机 演示主题交换机 项目介绍 此处我们模拟 RabbitMQ 实现了一个消息队列服务器 核心功能 提供了 虚拟主机、交换机、队列、绑定、消息 概念的管理九大核心 API 创建队列、销毁队列、创建交换机、销毁交换机、…

能碳双控| AIRIOT智慧能碳管理解决方案

在当前全球气候变化和可持续发展的背景下&#xff0c;建设能碳管理平台成为组织迎接挑战、提升可持续性的重要一环&#xff0c;有助于组织实现可持续发展目标&#xff0c;提高社会责任形象&#xff0c;同时适应未来碳排放管理的挑战。能碳管理是一个涉及跟踪、报告和减少组织碳…

pikachu靶场-RCE

介绍&#xff1a; RCE(remote command/code execute)概述 RCE漏洞&#xff0c;可以让攻击者直接向后台服务器远程注入操作系统命令或者代码&#xff0c;从而控制后台系统。 远程系统命令执行 一般出现这种漏洞&#xff0c;是因为应用系统从设计上需要给用户提供指定的远程命…

详解顺序结构滑动窗口处理算法

&#x1f380;个人主页&#xff1a; https://zhangxiaoshu.blog.csdn.net &#x1f4e2;欢迎大家&#xff1a;关注&#x1f50d;点赞&#x1f44d;评论&#x1f4dd;收藏⭐️&#xff0c;如有错误敬请指正! &#x1f495;未来很长&#xff0c;值得我们全力奔赴更美好的生活&…

Another Redis Desktop Manager工具连接集群

背景&#xff1a;使用Another Redis Desktop Manager连接redsi集群 win10安装 使用 下载 某盘&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1dg9kPm9Av8-bbpDfDg9DsA 提取码&#xff1a;t1sm 使用

(っ•̀ω•́)っ 如何在PPT中为文本框添加滚动条

本人在写技术分享的PPT时&#xff0c;遇到问题&#xff1a;有一大篇的代码&#xff0c;如何在一张PPT页面上显示&#xff1f;急需带有滚动条的文本框&#xff01;百度了不少&#xff0c;自己也来总结一篇&#xff0c;如下&#xff1a; 1、找到【文件】-【选项】 2、【自定义功…

Java面试笔记

Java面试笔记 Java面试笔记-网络模块 TCP的三次握手 TCP的简介&#xff1a; 面向连接的、可靠的、基于字节流的传输层通信协议 将应用层的数据流分割成报文段并发送给目标节点的TCP层 数据包都有序号&#xff0c;对方收到则发送ACK确认&#xff0c;未收到则重传 使用校验和来…

嵌入式Qt 实现用户界面与业务逻辑分离

一.基本程序框架一般包含 二.框架的基本设计原则 三.用户界面与业务逻辑的交互 四.代码实现计算器用户界面与业务逻辑 ICalculator.h #ifndef _ICALCULATOR_H_ #define _ICALCULATOR_H_#include <QString>class ICalculator { public:virtual bool expression(const QSt…
推荐文章