NFT交易市场开发(一)

实现的基本功能

  • (一)
    • 发行一个符合ERC20标准的测试Token,要求如下:
      • 总量::1亿
      • 精度:18
      • 名称:Fake USDT in CBI
      • 简称:cUSDT
  • (二)
    • 发行一个符合ERC721标准的测试Token,要求如下:
      • 名称:NFTMarketplace
      • 简称:NFTM
      • tokenid自增
      • 用户在网页上传图片提交之后自动发行给用户
  • (三)
    • 开发一个NFT交易市场,功能如下:
      • 用户在创建NFT时可以指定价格(以cUSDT计价)
      • NFT所有者可以修改上架的NFFT价格
      • NFT信息上传到IPFS
      • NFT所有者可以下架市场里自己的NFT
      • 用户可以在NFT市场以一定的价格购买NFT

实现的详细步骤

初始化hardhat工程
  • 前提是要安装好node.js。
  • 在自己新建的项目目录下打开终端
  • 使用npm install --save-dev hardhat命令安装hardhat
  • 输入npx hardhat init命令进行初始化
    在这里插入图片描述
  • 然后就搭建好了我们的一个hardhat框架
    在这里插入图片描述

ERC20

  • 然后输入code .命令在vscode打开该文件夹,在contracts文件夹下新建一个erc20-usdt.sol的合约文件,该文件夹下的初始合约文件可以删除,然后打开https://wizard.openzeppelin.com/#erc721网址复制合约代码
    在这里插入图片描述

  • 然后在vscode按照开发要求修改代码,修改后代码如下
    在这里插入图片描述

  • 在终端输入npm install @openzeppelin/contracts命令安装第三方的库

  • 在remix里面部署合约测试,关于本地文件连接remix教程前面文章详细讲过
    在这里插入图片描述

ERC721

  • 还是从之前的网址复制代码
    在这里插入图片描述
  • 新建erc721-nft.sol合约文件,将代码复制到该文件,然后编译测试一下

NFT交易市场

  • 新建一个nft-market.sol文件
  • 根据要求写合约代码
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.20;import "@openzeppelin/contracts/interfaces/IERC20.sol";
import "@openzeppelin/contracts/interfaces/IERC721.sol";contract Market {//将erc20初始化为一个IERC20类型的实例,但具体的地址尚未赋值,后续的构造函数会使用传入的参数来设置erc20,从而与实际的ERC20代币合约进行交互IERC20 public erc20;IERC721 public erc721;bytes4 internal constant MAGIC_ON_ERC721_RECEIVED = 0x150b7a02;//包含代币售卖地址,代币Id,代币价格struct Order {address seller;uint256 tokenId;uint256 price;}mapping(uint256 => Order) public orderOfId;   //根据tokenId查询OrderOrder[] public orders;   //定义一个Order类型的数组存放所有的Ordermapping (uint256 => uint256) public idToOrderIndex;   //根据tokenId查询在数组中的索引event Deal(address seller, address buyer, uint256 tokenId, uint256 price);  //交易信息event NewOrder(address seller, uint256 tokenId, uint256 price);  //新的卖家信息event PriceChanged(address seller, uint256 tokenId, uint256 previousPrice, uint256 price);  //价格修改信息event OrderCancelled(address seller, uint256 tokenId);    //卖家退出交易市场信息//获取ERC20和ERC721的合约地址,这俩合约地址在部署合约时,在详细信息里面可以找到,合约地址不能时零地址constructor(address _erc20, address _erc721) {require(_erc20 != address(0), "zero address");require(_erc721 != address(0), "zero address");//将传入的地址_erc20转换为IERC20类型的实例,并将其赋值给变量erc20erc20 = IERC20(_erc20);erc721 = IERC721(_erc721);}//购买函数function buy(uint256 _tokenId) external {//获取该tokenId的卖家地址address seller = orderOfId[_tokenId].seller;//获取该tokenI的售卖价格uint256 price = orderOfId[_tokenId].price;//买方地址即当前调用合约的地址address buyer = msg.sender;//调用ERC20里面的授权转账函数进行转账require(erc20.transferFrom(buyer, seller, price), "transfer not successful");//调用了ERC721代币合约的safeTransferFrom函数,将指定的ERC721代币(由_tokenId标识)从当前合约地址(address(this))安全地转移到买家账户(buyer)。erc721.safeTransferFrom(address(this), buyer, _tokenId);emit Deal(seller, buyer, _tokenId, price);  //释放交易信息}//下架函数function cancelOrder(uint256 _tokenId) external {//获取卖家地址address seller = orderOfId[_tokenId].seller;//查看当前地址是不是卖家,因为商品只能由卖家自己下架require(msg.sender == seller, "not seller");erc721.safeTransferFrom(address(this),seller, _tokenId);emit OrderCancelled(seller, _tokenId);}//改价函数function changePrice(uint256 _tokenId, uint256 _price) external {address seller = orderOfId[_tokenId].seller;require(msg.sender == seller, "not seller");uint256 previousPrice = orderOfId[_tokenId].price;orderOfId[_tokenId].price = _price;//注意,不仅要修改Order的价格,orders数组里面存储的该地址的代币价格也需要修改//memory修改不会存储在链上,用于存储临时数据,storage修改会存储在链上,用于永久保存 Order storage order = orders[idToOrderIndex[_tokenId]];order.price = _price;emit PriceChanged(seller, _tokenId, previousPrice, _price);}//上架函数//用于处理ERC721代币的接收事件,当其它合约或用户向该合约发送ERC721代币时,会触发该函数,可以在该函数中编写逻辑来处理接收到的代币function onERC721Received(address operator,   //执行操作的地址address from,    //发送代币的地址uint256 tokenId,   //接收到的代币的唯一标识符bytes calldata data  //附加数据,可以是任意字节数组) external returns (bytes4) {//调用格式转换函数得到价格uint256 price= toUint256(data, 0);require(price > 0, "price must be greater than 0");//上架orders.push(Order(from, tokenId, price));orderOfId[tokenId] = Order(from, tokenId, price);idToOrderIndex[tokenId] = orders.length - 1;emit NewOrder(from, tokenId, price);//返回该值,这样发送方的合约就可以指导接收方是否已经正确处理了代币转移return MAGIC_ON_ERC721_RECEIVED;}//下架函数function removeOrder(uint256 _tokenId) internal {//先从数组中把他删除掉uint256 index = idToOrderIndex[_tokenId];uint256 lastIndex = orders.length - 1;if(index != lastIndex){Order storage lastOrder = orders[lastIndex];orders[index] = lastOrder;idToOrderIndex[lastOrder.tokenId] = index;}orders.pop();//从mapping里面删除delete  orderOfId[_tokenId];delete idToOrderIndex[_tokenId];}//格式转换函数function toUint256(bytes memory _bytes, uint256 _start) public pure returns (uint256) {require(_start + 32 >= _start,"Market: toUint256_overflow");require(_bytes.length >= _start + 32,"Market: toUint256_outOfBounds");uint256 tempUint;assembly {tempUint := mload(add(add(_bytes, 0x20), _start))}return tempUint;}//获取数组订单长度function getOrderLength() external view returns (uint256) {return orders.length;}//获取所有上架的NFTfunction getAllNFTs() external view returns (Order[] memory) {return orders;}//获取用户自己上架的NFTfunction getMyNFTs() external view returns (Order[] memory) {Order[] memory myOrders = new Order[](orders.length);uint256 count = 0;for (uint256 i = 0; i< orders.length; i++) {if (orders[i].seller == msg.sender) {myOrders[count] = orders[i];count ++;}}return myOrders;}
}

编译测试

  • 先编译部署ERC20,这个直接编译部署就行
  • 然后编译部署ERC721
    在这里插入图片描述
  • 编译部署NFT合约
    在这里插入图片描述
  • 在ERC721部署的合约里给自己mint几个NFT
    在这里插入图片描述
    在这里插入图片描述
  • 查看是否mint成功,我们可以看到地址是我们刚刚mint的地址,成功!
  • 上架!NFT价格参数格式在https://stackoverflow.com/questions/63252057/how-to-use-bytestouint-function-in-solidity-the-one-with-assembly这个里面找

    在这里插入图片描述
  • 在nft合约里验证是否上架
  • 再去ERC721里面上架代币,一样的,修改下tokenId就可以,因为在ERC721里面代币是唯一的,所以每个代币的tokenId都不同
    在这里插入图片描述
  • 获取所有上架的NFT信息
    在这里插入图片描述
  • 用户获取自己上架了几个NFT,在nft合约里面查看
    在这里插入图片描述
  • 初步已经实现了,后续还有,课程可以去哔哩哔哩上搜梁培利,很好的老师,也是我的老师。

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

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

相关文章

学习方法 学习态度

学习方法 弄清它的历史、局限性和本质 常用的:刻意练习、熟能生巧 不常用的:知道在哪里找就行 事有先后&#xff0c;物有次序&#xff0c;盖房屋必须从平地建起 学习态度 每天比前一天的自己进步一点点

数据结构——算法的空间复杂度

【本节内容】 1.空间复杂度 2.常见空间复杂度 1.空间复杂度 空间复杂度也是一个数学表达式&#xff0c;是对一个算法在运行过程中临时占用额外存储空间大小的量度。 空间复杂度不是程序占用了多少bytes的空间&#xff0c;因为这个也没太大意义&#xff0c;所以空间复杂度算…

random标准模块

一、概述 在 Python 中&#xff0c;random 是一个内置模块&#xff0c;用于生成随机数。它提供了各种用于生成随机数的函数&#xff0c;包括伪随机数生成器、随机序列操作等。 1、需要导包 不会自动导入&#xff0c;需要显示的将random模块导入 import random2、源码分析&…

课时59:流程控制_if条件控制_语法解读

2.2.1 语法解读 学习目标 这一节&#xff0c;我们从 基础知识、简单实践、小结 三个方面来学习。 基础知识 简介 条件结构能够根据某个特定的条件&#xff0c;结合内置的测试表达式功能&#xff0c;结合测试的结果状态值对于条件进行判断&#xff0c;然后选择执行合适的任务…

Android开发学习-内容共享ContentProvoder(server+client)

在应用之间共享数据 通过ContentProvider封装数据 ContentProvider使用的Uri语法结构如下&#xff1a; content://authority/data_path/id content&#xff1a;通用前缀&#xff0c;表示该uri用于ContentProvider定位资源 authority&#xff1a;是授权者名称&#xff0c;用…

【Greenhills】MULTIIDE集成第三方的编辑器进行源文件编辑工作

【更多软件使用问题请点击亿道电子官方网站查询】 1、 文档目标 在使用GHS进行工作的时候&#xff0c;可以集成第三方的编辑器进行源文件编辑工作 2、 问题场景 用于解决在GHS中进行项目开发时&#xff0c;对于GHS的编辑器使用不习惯&#xff0c;想要切换到其他第三方的编辑…

基于c语言的大宗商品撮合交易平台的市场价值

大宗商品撮合交易平台的市场价值主要体现在以下几个方面&#xff1a; 提高市场流动性&#xff1a;平台通过自动化撮合和高效的交易处理&#xff0c;降低了交易成本&#xff0c;提高了市场流动性。这使得投资者能够更容易地找到合适的交易对手&#xff0c;促进交易的成交。 促进…

以题为例 浅谈sql注入布尔盲注

什么是布尔盲注 布尔盲注就是在注入过程中&#xff0c;页面只会返回false和true&#xff0c;不会去返回其它的信息&#xff0c;所以我们不能通过语句查询直接获得数据库的名字&#xff0c;而是通过逻辑获得数据库的信息 布尔盲注常使用函数 length() 返回字符串的长度&#…

TDengine 签约益和热力,应对智慧供热系统大规模数据挑战

近日&#xff0c;涛思数据与安阳益和热力集团有限公司达成合作协议&#xff0c;共同致力于推动智慧供热系统的建设和发展。作为安阳市城市集中供热的主要负责单位&#xff0c;益和热力将借助涛思数据先进的技术和解决方案&#xff0c;提升供热行业的管理效率和服务质量。在本次…

the demo for C# multicast delegate 多播委托

委托类Delegate&#xff0c;位于System命名空间下,是所有声明的委托类型的基类. 例如我们声明了一个委托类型Func<int,bool> MyFunc或者 delegate bool MyDel(int i); 该类型声明成功后&#xff0c;该委托类自动继承System.MulticastDelegate&#xff0c;其包含了构造…

为什么在镀膜时要测薄膜折射率?

在芯片制造中&#xff0c;镀膜工序&#xff08;PVD,CVD&#xff09;是必不可少的关键环节&#xff0c;薄膜的质量直接影响了芯片的性能。对这些薄膜的精细控制又离不开对其折射率的深入理解和精确测量。今天将对芯片制造中薄膜折射率的概念、测量方法&#xff0c;以及它在整个制…

Arcgis小技巧【19】——更改字段顺序

一、问题分析 一般情况下&#xff0c;一个合格且严谨的要素或表数据&#xff0c;它的字段顺序都是固定的。 比如三调数据&#xff0c;正常的字段数据如下&#xff08;截取部分字段&#xff09;&#xff1a; 数据经过处理&#xff0c;如增删字段&#xff0c;可能会出现字段顺序…

SAM模型

SAM与过去分割对比 根据以下Demo,我们可以发现&#xff0c;通过在图像中指定要分割的内容提示&#xff0c;SAM可以实现各种分割任务&#xff0c;且无需额外的训练、做到零样本泛化&#xff0c;即SAM学会了辨别物体、具备图像理解力、对不熟悉的图像和物体能进行零样本概括&…

Hyperf AOP 和 注解

注解 (hyperf.wiki) AOP 面向切面编程 (hyperf.wiki) 切面 定义切面(Aspect) 根据官方教程定义一个切面。可以指定类、方法、参数和注解上生效。 <?php namespace App\Aspect;use App\Service\SomeClass; use App\Annotation\SomeAnnotation; use Hyperf\Di\Annotatio…

论文阅读《FENET: FOCUSING ENHANCED NETWORK FOR LANE DETECTION》

ABSTRACT 受人类驾驶专注力的启发&#xff0c;这项研究开创性地利用聚焦采样&#xff08;Focusing Sampling&#xff09;、部分视野评估&#xff08;Partial Field of View Evaluation&#xff09;、增强型 FPN 架构和定向 IoU 损失&#xff08;Directional IoU Loss&#xff…

STM32单片机示例:ETH_LAN8742_DHCP_NonOS_Poll_H743

文章目录 目的基础说明关键配置关键代码示例链接总结 目的 以太网是比较常用到的功能&#xff0c;STM32系列单片机使用CubeMX配置使用以太网功能比非常方便。不过对于H7系列来说需要使能 DCache 才能启用LwIP&#xff0c;启用Cache后又会带来一些需要特别注意的事情。这篇文章…

急速建立网站方法

急速建立网站方法 WordPress&#xff1a; 简介&#xff1a; WordPress是一种广泛用于建设博客、网站的免费开源内容管理系统&#xff08;CMS&#xff09;。它具有友好的用户界面和丰富的插件生态系统&#xff0c;使得用户可以轻松创建和管理网站。特点&#xff1a; 主题和插件支…

AI智能应用百科立即落地实操课

该课程旨在教授学员如何将AI智能应用于实际场景。通过深入的案例研究和实操练习&#xff0c;学员将学会应用机器学习、自然语言处理等技术&#xff0c;快速解决现实问题。课程强调实际操作&#xff0c;帮助学员快速运用AI技术解决工作中的挑战。 课程大小&#xff1a;3.3G 课…

PyTorch会在每次.backward()调用时会累积梯度的问题

代码 # backward() accumulates the gradient for this tensor into .grad attribute. # !!! We need to be careful during optimization !!! # Use .zero_() to empty the gradients before a new optimization step! weights torch.ones(4, requires_gradTrue)for epoch i…

10 | MySQL为什么有时候会选错索引?

前面我们介绍过索引&#xff0c;你已经知道了在 MySQL 中一张表其实是可以支持多个索引的。但是&#xff0c;你写 SQL 语句的时候&#xff0c;并没有主动指定使用哪个索引。也就是说&#xff0c;使用哪个索引是由 MySQL 来确定的。 不知道你有没有碰到过这种情况&#xff0c;一…