DAPP开发【05】ERC20/ERC721简介

2023-12-03 12:15
文章标签 开发 05 简介 dapp erc721 erc20

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

ERC20是以太坊上最受欢迎和广泛使用的代币标准之一。ERC20代币符合一组通用规则,包括代币的转账方法和余额查询方法。这些规则使得 ERC20 代币易于与钱包、交易所和其他合约进行集成和交互。

ERC20标准规定了代币合约必须实现以下6个函数:

  1. balanceOf(address _owner):获取给定地址的代币余额。

  2. transfer(address _to, uint256 _value):将代币从一个地址转移到另一个地址。

  3. allowance(address _owner, address _spender):返回给定地址的可转移余额。

  4. approve(address _spender, uint256 _value):允许另一个地址花费指定数量的代币。

  5. transferFrom(address _from, address _to, uint256 _value):从一个地址转移代币到另一个地址,前提是交易需要经过第三方的批准。

  6. totalSupply():获取代币的总供应量。

ERC20代币合约必须遵守并实现这些函数,以确保它们与其他代币和钱包兼容。此外,ERC20代币合约还必须包含代币的名称、符号和小数位数等元数据。

任何人都可以创建自己的ERC20代币,并将其用于支付、投票或其他用途。同时, ERC20代币也很容易与其他代币和钱包进行交互。

ERC20代币具有以下属性:

  1. 总供应量(Total Supply): 该代币发行的总数量。

  2. 名称(Name): 代币的名称,例如“以太币”。

  3. 符号(Symbol): 代币的简称,例如“ETH”。

  4. 小数位数(Decimals): 代币可以拆分的最小单位,例如“18”,代表代币可以被拆分为10的18次方的最小单位。

  5. 余额(Balance): 代币持有人的余额。

  6. 转移记录(Transfer Record): 代币的转移记录,包括发送、接收方、数量和时间。

  7. 批准记录(Approval Record): 代币的批准记录,包括批准者、受批准者和数量。

  8. 总供应限制(Total Supply Limit): 可以控制代币的总供应量,可以选择是否限制代币的总供应量。

以上属性可以在ERC20代币合约中进行定义和设置,以确保标准化和互操作性。

ERC721是一种以太坊上的代币标准,用于创建不可替代的代币。与ERC20代币不同,每个ERC721代币都是独一无二的,因此它们通常用于代表唯一的资产,如游戏中的虚拟物品、数字艺术品、房屋和土地证书等。

ERC721代币具有以下属性:

  1. 每个ERC721代币都有一个独一无二的标识符(Token ID),通过这个标识符可以唯一地标识每个代币。

  2. ERC721代币不可替代,即每个代币都是唯一的,不能被其他代币所替代。

  3. 每个ERC721代币都可以被拥有者进行转移,就像普通的代币一样。

  4. ERC721合约必须实现一组标准接口,包括balanceOfownerOfapprovetransferFromsafeTransferFrom,以便于与其他合约进行交互和转移代币。

  5. ERC721代币可以被继承和扩展,以实现更复杂的功能,例如用于游戏的虚拟物品。

下面是一个以太坊智能合约的示例代码,用于实现一个简单的ERC721代币合约:

pragma solidity ^0.8.0;contract MyNFT {// ERC721 标准string public constant name = "My NFT";string public constant symbol = "MYNFT";event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);mapping(uint256 => address) private _tokenOwner;mapping(uint256 => address) private _tokenApprovals;mapping(address => uint256) private _ownedTokensCount;function balanceOf(address owner) public view returns (uint256) {require(owner != address(0), "ERC721: balance query for the zero address");return _ownedTokensCount[owner];}function ownerOf(uint256 tokenId) public view returns (address) {address owner = _tokenOwner[tokenId];require(owner != address(0), "ERC721: owner query for nonexistent token");return owner;}function approve(address to, uint256 tokenId) public {address owner = ownerOf(tokenId);require(to != owner, "ERC721: approval to current owner");require(msg.sender == owner || isApprovedForAll(owner, msg.sender),"ERC721: approve caller is not owner nor approved for all");_tokenApprovals[tokenId] = to;emit Approval(owner, to, tokenId);}function getApproved(uint256 tokenId) public view returns (address) {require(_exists(tokenId), "ERC721: approved query for nonexistent token");return _tokenApprovals[tokenId];}function setApprovalForAll(address operator, bool approved) public {require(operator != msg.sender, "ERC721: approve to caller");_operatorApprovals[msg.sender][operator] = approved;emit ApprovalForAll(msg.sender, operator, approved);}function isApprovedForAll(address owner, address operator) public view returns (bool) {return _operatorApprovals[owner][operator];}function transferFrom(address from, address to, uint256 tokenId) public {require(_isApprovedOrOwner(msg.sender, tokenId), "ERC721: transfer caller is not owner nor approved");_transfer(from, to, tokenId);}function safeTransferFrom(address from, address to, uint256 tokenId) public {safeTransferFrom(from, to, tokenId, "");}function safeTransferFrom(address from, address to, uint256 tokenId, bytes memory _data) public {require(_isApprovedOrOwner(msg.sender, tokenId), "ERC721: transfer caller is not owner nor approved");_safeTransfer(from, to, tokenId, _data);}// 扩展功能function mint(address to, uint256 tokenId) public {require(to != address(0), "ERC721: mint to the zero address");require(!_exists(tokenId), "ERC721: token already minted");_tokenOwner[tokenId] = to;_ownedTokensCount[to] += 1;emit Transfer(address(0), to, tokenId);}function _exists(uint256 tokenId) internal view returns (bool) {address owner = _tokenOwner[tokenId];return owner != address(0);}function _isApprovedOrOwner(address spender, uint256 tokenId) internal view returns (bool) {require(_exists(tokenId), "ERC721: operator query for nonexistent token");address owner = ownerOf(tokenId);return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender));}function _safeTransfer(address from, address to, uint256 tokenId, bytes memory _data) internal {_transfer(from, to, tokenId);require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer");}function _transfer(address from, address to, uint256 tokenId) internal {require(ownerOf(tokenId) == from, "ERC721: transfer of token that is not own");require(to != address(0), "ERC721: transfer to the zero address");_tokenApprovals[tokenId] = address(0);_ownedTokensCount[from] -= 1;_ownedTokensCount[to] += 1;_tokenOwner[tokenId] = to;emit Transfer(from, to, tokenId);}function _checkOnERC721Received(address from, address to, uint256 tokenId, bytes memory _data) internal returns (bool) {if (isContract(to)) {try IERC721Receiver(to).onERC721Received(msg.sender, from, tokenId, _data) returns (bytes4 retval) {return retval == IERC721Receiver(to).onERC721Received.selector;} catch (bytes memory reason) {if (reason.length == 0) {revert("ERC721: transfer to non ERC721Receiver implementer");} else {assembly {revert(add(32, reason), mload(reason))}}}} else {return true;}}function isContract(address account) internal view returns (bool) {uint256 size;assembly { size := extcodesize(account) }return size > 0;}
}interface IERC721Receiver {function onERC721Received(address operator,address from,uint256 tokenId,bytes calldata data) external returns (bytes4);
}

这个示例代码中包含了ERC721代币标准的主要功能,包括标识符、转移、批准、所有者等。此外,它还包括了一些扩展功能,例如 mint 函数用于发行新的代币。请注意,这个示例代码并不完整,你需要根据自己的需求进行修改和扩展。

这篇关于DAPP开发【05】ERC20/ERC721简介的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python38个游戏开发库整理汇总

《Python38个游戏开发库整理汇总》文章介绍了多种Python游戏开发库,涵盖2D/3D游戏开发、多人游戏框架及视觉小说引擎,适合不同需求的开发者入门,强调跨平台支持与易用性,并鼓励读者交流反馈以... 目录PyGameCocos2dPySoyPyOgrepygletPanda3DBlenderFife

使用Python开发一个Ditto剪贴板数据导出工具

《使用Python开发一个Ditto剪贴板数据导出工具》在日常工作中,我们经常需要处理大量的剪贴板数据,下面将介绍如何使用Python的wxPython库开发一个图形化工具,实现从Ditto数据库中读... 目录前言运行结果项目需求分析技术选型核心功能实现1. Ditto数据库结构分析2. 数据库自动定位3

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

PyQt5 GUI 开发的基础知识

《PyQt5GUI开发的基础知识》Qt是一个跨平台的C++图形用户界面开发框架,支持GUI和非GUI程序开发,本文介绍了使用PyQt5进行界面开发的基础知识,包括创建简单窗口、常用控件、窗口属性设... 目录简介第一个PyQt程序最常用的三个功能模块控件QPushButton(按钮)控件QLable(纯文本

基于Python开发一个图像水印批量添加工具

《基于Python开发一个图像水印批量添加工具》在当今数字化内容爆炸式增长的时代,图像版权保护已成为创作者和企业的核心需求,本方案将详细介绍一个基于PythonPIL库的工业级图像水印解决方案,有需要... 目录一、系统架构设计1.1 整体处理流程1.2 类结构设计(扩展版本)二、核心算法深入解析2.1 自

Qt QCustomPlot库简介(最新推荐)

《QtQCustomPlot库简介(最新推荐)》QCustomPlot是一款基于Qt的高性能C++绘图库,专为二维数据可视化设计,它具有轻量级、实时处理百万级数据和多图层支持等特点,适用于科学计算、... 目录核心特性概览核心组件解析1.绘图核心 (QCustomPlot类)2.数据容器 (QCPDataC

SpringBoot开发中十大常见陷阱深度解析与避坑指南

《SpringBoot开发中十大常见陷阱深度解析与避坑指南》在SpringBoot的开发过程中,即使是经验丰富的开发者也难免会遇到各种棘手的问题,本文将针对SpringBoot开发中十大常见的“坑... 目录引言一、配置总出错?是不是同时用了.properties和.yml?二、换个位置配置就失效?搞清楚加

Python中对FFmpeg封装开发库FFmpy详解

《Python中对FFmpeg封装开发库FFmpy详解》:本文主要介绍Python中对FFmpeg封装开发库FFmpy,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录一、FFmpy简介与安装1.1 FFmpy概述1.2 安装方法二、FFmpy核心类与方法2.1 FF

基于Python开发Windows屏幕控制工具

《基于Python开发Windows屏幕控制工具》在数字化办公时代,屏幕管理已成为提升工作效率和保护眼睛健康的重要环节,本文将分享一个基于Python和PySide6开发的Windows屏幕控制工具,... 目录概述功能亮点界面展示实现步骤详解1. 环境准备2. 亮度控制模块3. 息屏功能实现4. 息屏时间