摘要:它和我寫的上一篇源碼分析介紹的有所不同,最小的單位為無法再分割,代表獨一無二的,針對不可置換的的智能合約標準接口。源碼分析到這里就結束了。
ERC721 官方簡介是:A standard interface for non-fungible tokens, also known as deeds.也叫非同質代幣,或者不可置換代幣(NFTs)。提到ERC721,一個好理解的例子就是CryptoKitties 迷戀貓,每一只貓都是獨一無二的擁有不同基因,有收藏價值屬性。ERC721對于虛擬資產收藏品領域會有很好的應用價值和市場需求。
它和我寫的上一篇《OpenZeppelin ERC20源碼分析》介紹的ERC20有所不同,ERC721最小的單位為1無法再分割,代表獨一無二的,針對不可置換的Token的智能合約標準接口。從 ERC721標準草案中可以看到,兼容ERC20的方法有4個:name,symbol,totalSupply,balanceOf 添加的新方法:ownerOf,takeOwnership ERC721還重寫了approve和transfer。
分析OpenZeppelin ERC721源碼前同樣我畫了一個繼承和調用關系的思維導圖,可以幫助更容易地看源碼。
ERC721Basic.solpragma solidity ^0.4.23; /** * @title ERC721 標準的基本接口 * @dev see https://github.com/ethereum/EIPs/blob/master/EIPS/eip-721.md */ contract ERC721Basic { event Transfer( address indexed _from, address indexed _to, uint256 _tokenId ); event Approval( address indexed _owner, address indexed _approved, uint256 _tokenId ); event ApprovalForAll( address indexed _owner, address indexed _operator, bool _approved ); function balanceOf(address _owner) public view returns (uint256 _balance); function ownerOf(uint256 _tokenId) public view returns (address _owner); function exists(uint256 _tokenId) public view returns (bool _exists); function approve(address _to, uint256 _tokenId) public; function getApproved(uint256 _tokenId) public view returns (address _operator); function setApprovalForAll(address _operator, bool _approved) public; function isApprovedForAll(address _owner, address _operator) public view returns (bool); function transferFrom(address _from, address _to, uint256 _tokenId) public; function safeTransferFrom(address _from, address _to, uint256 _tokenId) public; function safeTransferFrom( address _from, address _to, uint256 _tokenId, bytes _data ) public; }
ERC721Basic 合約定義了基本的接口方法:
balanceOf 返回_owner的代幣數量
ownerOf 根據_tokenId返回代幣持有者address
exists _tokenId是否存在
approve 授權_tokenId給地址to
getApproved 查詢_tokenId的授權人_operator address
setApprovalForAll 授權_operator具有所有代幣的控制權
isApprovedForAll
transferFrom 轉移代幣所有權
safeTransferFrom 轉移代幣所有權
同時還定義了Transfer Approval ApprovalForAll 在后面的ERC721實現的代碼中再來看事件的觸發。
ERC721.solpragma solidity ^0.4.23; import "./ERC721Basic.sol"; /** * @title ERC-721 標準的基本接口, 可選的枚舉擴展 * @dev See https://github.com/ethereum/EIPs/blob/master/EIPS/eip-721.md */ contract ERC721Enumerable is ERC721Basic { function totalSupply() public view returns (uint256); function tokenOfOwnerByIndex( address _owner, uint256 _index ) public view returns (uint256 _tokenId); function tokenByIndex(uint256 _index) public view returns (uint256); } /** * @title ERC-721 ERC-721 標準的基本接口, 可選的元數據擴展 * @dev See https://github.com/ethereum/EIPs/blob/master/EIPS/eip-721.md */ contract ERC721Metadata is ERC721Basic { function name() public view returns (string _name); function symbol() public view returns (string _symbol); function tokenURI(uint256 _tokenId) public view returns (string); } /** * @title ERC-721 標準的基本接口,完整實現接口 * @dev See https://github.com/ethereum/EIPs/blob/master/EIPS/eip-721.md */ contract ERC721 is ERC721Basic, ERC721Enumerable, ERC721Metadata { }
ERC721 合約繼承了 ERC721Basic 的基礎上,添加枚舉和元數據的擴展。
ERC721Enumerable枚舉擴展可以使代幣更具有可訪問性:
totalSupply 返回代幣總量
tokenOfOwnerByIndex 通過_owner所有者地址和索引值返回所有者代幣列表中的_tokenId
tokenByIndex 通過索引值返回tokenId
ERC721Metadata元數據擴展哦用來描述合約元信息
name 返回合約名字
symbol 返回代幣符號
tokenURI 返回_tokenId對應的資源URI
ERC721BasicTokenpragma solidity ^0.4.23; import "./ERC721Basic.sol"; import "./ERC721Receiver.sol"; import "../../math/SafeMath.sol"; import "../../AddressUtils.sol"; /** * @title ERC721 標準基本實現 * @dev see https://github.com/ethereum/EIPs/blob/master/EIPS/eip-721.md */ contract ERC721BasicToken is ERC721Basic { using SafeMath for uint256; using AddressUtils for address; // Equals to `bytes4(keccak256("onERC721Received(address,uint256,bytes)"))` // which can be also obtained as `ERC721Receiver(0).onERC721Received.selector` bytes4 constant ERC721_RECEIVED = 0xf0b9e5ba; // token ID 到 持有人owner的映射 mapping (uint256 => address) internal tokenOwner; // token ID 到授權地址address的映射 mapping (uint256 => address) internal tokenApprovals; // 持有人到持有的token數量的映射 mapping (address => uint256) internal ownedTokensCount; // 持有人到操作人授權的映射 mapping (address => mapping (address => bool)) internal operatorApprovals; /** * @dev 確保msg.sender是tokenId的持有人 * @param _tokenId uint256 ID of the token to validate its ownership belongs to msg.sender */ modifier onlyOwnerOf(uint256 _tokenId) { require(ownerOf(_tokenId) == msg.sender); _; } /** * @dev 通過檢查msg.sender是否是代幣的持有人,被授權或者操作人來確保msg.sender可以交易一個token * @param _tokenId uint256 ID of the token to validate */ modifier canTransfer(uint256 _tokenId) { require(isApprovedOrOwner(msg.sender, _tokenId)); _; } /** * @dev 獲取持有者的代幣總數 * @param _owner address to query the balance of * @return uint256 representing the amount owned by the passed address */ function balanceOf(address _owner) public view returns (uint256) { require(_owner != address(0)); return ownedTokensCount[_owner]; } /** * @dev 根據token ID獲取持有者 * @param _tokenId uint256 ID of the token to query the owner of * @return owner address currently marked as the owner of the given token ID */ function ownerOf(uint256 _tokenId) public view returns (address) { address owner = tokenOwner[_tokenId]; require(owner != address(0)); return owner; } /** * @dev 指定的token是否存在 * @param _tokenId uint256 ID of the token to query the existence of * @return whether the token exists */ function exists(uint256 _tokenId) public view returns (bool) { address owner = tokenOwner[_tokenId]; return owner != address(0); } /** * @dev 批準另一個人address來交易指定的代幣 * @dev 0 address 表示沒有授權的地址 * @dev 給定的時間內,一個token只能有一個批準的地址 * @dev 只有token的持有者或者授權的操作人才可以調用 * @param _to address to be approved for the given token ID * @param _tokenId uint256 ID of the token to be approved */ function approve(address _to, uint256 _tokenId) public { address owner = ownerOf(_tokenId); require(_to != owner); require(msg.sender == owner || isApprovedForAll(owner, msg.sender)); if (getApproved(_tokenId) != address(0) || _to != address(0)) { tokenApprovals[_tokenId] = _to; emit Approval(owner, _to, _tokenId); } } /** * @dev 獲取token被授權的地址,如果沒有設置地址則為0 * @param _tokenId uint256 ID of the token to query the approval of * @return address currently approved for the given token ID */ function getApproved(uint256 _tokenId) public view returns (address) { return tokenApprovals[_tokenId]; } /** * @dev 設置或者取消對操作人的授權 * @dev 一個操作人可以代表他們轉讓發送者的所有token * @param _to operator address to set the approval * @param _approved representing the status of the approval to be set */ function setApprovalForAll(address _to, bool _approved) public { require(_to != msg.sender); operatorApprovals[msg.sender][_to] = _approved; emit ApprovalForAll(msg.sender, _to, _approved); } /** * @dev 查詢是否操作人被指定的持有者授權 * @param _owner 要查詢的授權人地址 * @param _operator 要查詢的授權操作人地址 * @return bool whether the given operator is approved by the given owner */ function isApprovedForAll( address _owner, address _operator ) public view returns (bool) { return operatorApprovals[_owner][_operator]; } /** * @dev 將指定的token所有權轉移給另外一個地址 * @dev 不鼓勵使用這個方法,盡量使用`safeTransferFrom` * @dev 要求 msg.sender 必須為所有者,已授權或者操作人 * @param _from current owner of the token * @param _to address to receive the ownership of the given token ID * @param _tokenId uint256 ID of the token to be transferred */ function transferFrom( address _from, address _to, uint256 _tokenId ) public canTransfer(_tokenId) { require(_from != address(0)); require(_to != address(0)); clearApproval(_from, _tokenId); removeTokenFrom(_from, _tokenId); addTokenTo(_to, _tokenId); emit Transfer(_from, _to, _tokenId); } /** * @dev 更安全的方法,將指定的token所有權轉移給另外一個地址 * @dev 如果目標地址是一個合約,必須實現 `onERC721Received`,這個要求安全交易并返回值 `bytes4(keccak256("onERC721Received(address,uint256,bytes)"))`; 否則交易被還原 * @dev 要求 msg.sender 必須為所有者,已授權或者操作人 * @param _from current owner of the token * @param _to address to receive the ownership of the given token ID * @param _tokenId uint256 ID of the token to be transferred */ function safeTransferFrom( address _from, address _to, uint256 _tokenId ) public canTransfer(_tokenId) { safeTransferFrom(_from, _to, _tokenId, ""); } /** * @dev 更安全的方法,將指定的token所有權轉移給另外一個地址 * @dev 如果目標地址是一個合約,必須實現 `onERC721Received`,這個要求安全交易并返回值 `bytes4(keccak256("onERC721Received(address,uint256,bytes)"))`; 否則交易被還原 * @dev 要求 msg.sender 必須為所有者,已授權或者操作人 * @param _from current owner of the token * @param _to address to receive the ownership of the given token ID * @param _tokenId uint256 ID of the token to be transferred * @param _data bytes data to send along with a safe transfer check */ function safeTransferFrom( address _from, address _to, uint256 _tokenId, bytes _data ) public canTransfer(_tokenId) { transferFrom(_from, _to, _tokenId); require(checkAndCallSafeTransfer(_from, _to, _tokenId, _data)); } /** * @dev 返回給定的spender是否可以交易一個給定的token * @param _spender address of the spender to query * @param _tokenId uint256 ID of the token to be transferred * @return bool whether the msg.sender is approved for the given token ID, * is an operator of the owner, or is the owner of the token */ function isApprovedOrOwner( address _spender, uint256 _tokenId ) internal view returns (bool) { address owner = ownerOf(_tokenId); return ( _spender == owner || getApproved(_tokenId) == _spender || isApprovedForAll(owner, _spender) ); } /** * @dev 增發一個新token的內部方法 * @dev 如果增發的token已經存在則撤銷 * @param _to The address that will own the minted token * @param _tokenId uint256 ID of the token to be minted by the msg.sender */ function _mint(address _to, uint256 _tokenId) internal { require(_to != address(0)); addTokenTo(_to, _tokenId); emit Transfer(address(0), _to, _tokenId); } /** * @dev 銷毀一個token的內部方法 * @dev 如果token不存在則撤銷 * @param _tokenId uint256 ID of the token being burned by the msg.sender */ function _burn(address _owner, uint256 _tokenId) internal { clearApproval(_owner, _tokenId); removeTokenFrom(_owner, _tokenId); emit Transfer(_owner, address(0), _tokenId); } /** * @dev 清除當前的給定token的授權,內部方法 * @dev 如果給定地址不是token的持有者則撤銷 * @param _owner owner of the token * @param _tokenId uint256 ID of the token to be transferred */ function clearApproval(address _owner, uint256 _tokenId) internal { require(ownerOf(_tokenId) == _owner); if (tokenApprovals[_tokenId] != address(0)) { tokenApprovals[_tokenId] = address(0); emit Approval(_owner, address(0), _tokenId); } } /** * @dev 內部方法,將給定的token添加到給定地址列表中 * @param _to address 指定token的新所有者 * @param _tokenId uint256 ID of the token to be added to the tokens list of the given address */ function addTokenTo(address _to, uint256 _tokenId) internal { require(tokenOwner[_tokenId] == address(0)); tokenOwner[_tokenId] = _to; ownedTokensCount[_to] = ownedTokensCount[_to].add(1); } /** * @dev 內部方法,將給定的token從地址列表中移除 * @param _from address 給定token的之前持有中地址 * @param _tokenId uint256 ID of the token to be removed from the tokens list of the given address */ function removeTokenFrom(address _from, uint256 _tokenId) internal { require(ownerOf(_tokenId) == _from); ownedTokensCount[_from] = ownedTokensCount[_from].sub(1); tokenOwner[_tokenId] = address(0); } /** * @dev 內部函數,調用目標地址上的 `onERC721Received` * @dev 如果目標地址不是合同則不執行調用 * @param _from address representing the previous owner of the given token ID * @param _to target address that will receive the tokens * @param _tokenId uint256 ID of the token to be transferred * @param _data bytes optional data to send along with the call * @return whether the call correctly returned the expected magic value */ function checkAndCallSafeTransfer( address _from, address _to, uint256 _tokenId, bytes _data ) internal returns (bool) { if (!_to.isContract()) { return true; } bytes4 retval = ERC721Receiver(_to).onERC721Received( _from, _tokenId, _data); return (retval == ERC721_RECEIVED); } }
ERC721BasicToken 實現了ERC721Basic合約定義的接口方法,主要對token的持有人的一個添加和修改,以及授權和交易的管理,實現了基本的非同質化token的業務邏輯。具體方法實現并不難,就是對映射的公有變量的管理,但是對于權限和安全驗證值得關注,比如函數修改器還有require。
ERC721Token.solpragma solidity ^0.4.23; import "./ERC721.sol"; import "./ERC721BasicToken.sol"; /** * @title 完整 ERC721 Token * 該實現包括所有ERC721標準必須的和可選的方法,此外還包括使用操作者批準所有功能 * @dev see https://github.com/ethereum/EIPs/blob/master/EIPS/eip-721.md */ contract ERC721Token is ERC721, ERC721BasicToken { // 代幣名稱 string internal name_; // 代幣符號 string internal symbol_; // 所有者到所有者擁有的代幣列表的映射 mapping(address => uint256[]) internal ownedTokens; // 所有者代幣列表中代幣ID到索引的映射 mapping(uint256 => uint256) internal ownedTokensIndex; // 保存所有代幣ID的數組,用于枚舉 uint256[] internal allTokens; // allTokens數組中代幣ID到索引的映射 mapping(uint256 => uint256) internal allTokensIndex; // 可選的代幣資源URIs映射 mapping(uint256 => string) internal tokenURIs; /** * @dev Constructor function */ constructor(string _name, string _symbol) public { name_ = _name; symbol_ = _symbol; } /** * @dev 獲取代幣名稱 * @return string representing the token name */ function name() public view returns (string) { return name_; } /** * @dev 獲取代幣符號 * @return string representing the token symbol */ function symbol() public view returns (string) { return symbol_; } /** * @dev 根據_tokenId返回對應的資源URI * @dev 如果token不存在異常返回空字符串 * @param _tokenId uint256 ID of the token to query */ function tokenURI(uint256 _tokenId) public view returns (string) { require(exists(_tokenId)); return tokenURIs[_tokenId]; } /** * @dev 獲取token id 通過給定的token列表中的索引 * @param _owner address owning the tokens list to be accessed * @param _index uint256 representing the index to be accessed of the requested tokens list * @return uint256 token ID at the given index of the tokens list owned by the requested address */ function tokenOfOwnerByIndex( address _owner, uint256 _index ) public view returns (uint256) { require(_index < balanceOf(_owner)); return ownedTokens[_owner][_index]; } /** * @dev 獲取合約存儲的token總數 * @return uint256 representing the total amount of tokens */ function totalSupply() public view returns (uint256) { return allTokens.length; } /** * @dev 根據token 索引值獲取合約中token的 * @dev 如果索引大于等于token總數則撤銷 * @param _index uint256 representing the index to be accessed of the tokens list * @return uint256 token ID at the given index of the tokens list */ function tokenByIndex(uint256 _index) public view returns (uint256) { require(_index < totalSupply()); return allTokens[_index]; } /** * @dev 內部方法,給存在token添加token URI * @dev Reverts if the token ID does not exist * @param _tokenId uint256 ID of the token to set its URI * @param _uri string URI to assign */ function _setTokenURI(uint256 _tokenId, string _uri) internal { require(exists(_tokenId)); tokenURIs[_tokenId] = _uri; } /** * @dev 內部方法,添加token ID 到給定的地址的列表中 * @param _to address 給定token ID的新的持有者 * @param _tokenId uint256 ID of the token to be added to the tokens list of the given address */ function addTokenTo(address _to, uint256 _tokenId) internal { // 調用父合約的addTokenTo super.addTokenTo(_to, _tokenId); uint256 length = ownedTokens[_to].length; ownedTokens[_to].push(_tokenId); //當前的長度作為索引 ownedTokensIndex[_tokenId] = length; } /** * @dev 內部方法,從一個給定地址的列表中移除token * @param _from address 給定token ID的之前的持有者address * @param _tokenId uint256 ID of the token to be removed from the tokens list of the given address */ function removeTokenFrom(address _from, uint256 _tokenId) internal { // 調用父合約的移除方法 super.removeTokenFrom(_from, _tokenId); // 獲取token的索引 uint256 tokenIndex = ownedTokensIndex[_tokenId]; // 獲取持有人token的最后一個token索引 uint256 lastTokenIndex = ownedTokens[_from].length.sub(1); // 獲取最后一個token uint256 lastToken = ownedTokens[_from][lastTokenIndex]; //將最后一個token放到被刪除的索引位置,lastTokenIndex置0 ownedTokens[_from][tokenIndex] = lastToken; ownedTokens[_from][lastTokenIndex] = 0; // 注意這里需要處理單元素數組,tokenIndex和lastTokenIndex都將置0.然后可以確保將ownedTokens列表中刪除_tokenId,首先將lastToken換到第一個位置,然后刪除列表最后位置的元素 ownedTokens[_from].length--; ownedTokensIndex[_tokenId] = 0; ownedTokensIndex[lastToken] = tokenIndex; } /** * @dev 內部方法,增發一個新的token * @dev 如果token已經存在了就撤銷 * @param _to address the beneficiary that will own the minted token * @param _tokenId uint256 ID of the token to be minted by the msg.sender */ function _mint(address _to, uint256 _tokenId) internal { super._mint(_to, _tokenId); allTokensIndex[_tokenId] = allTokens.length; allTokens.push(_tokenId); } /** * @dev 內部方法,銷毀一個指定的token * @dev token不存在則撤銷 * @param _owner owner of the token to burn * @param _tokenId uint256 ID of the token being burned by the msg.sender */ function _burn(address _owner, uint256 _tokenId) internal { super._burn(_owner, _tokenId); // 清除資源URI if (bytes(tokenURIs[_tokenId]).length != 0) { delete tokenURIs[_tokenId]; } // 做所有的token數組后續處理 uint256 tokenIndex = allTokensIndex[_tokenId]; uint256 lastTokenIndex = allTokens.length.sub(1); uint256 lastToken = allTokens[lastTokenIndex]; // 可以參考增發removeTokenFrom allTokens[tokenIndex] = lastToken; allTokens[lastTokenIndex] = 0; allTokens.length--; allTokensIndex[_tokenId] = 0; allTokensIndex[lastToken] = tokenIndex; } }
ERC721Token實現了完整的ERC721標準,在繼承了ERC721BasicToken的基礎上增加了一些token的操作,主要在包括token的元數據,資源URI,增發銷毀,還有就是token索引的映射關系。對于具體實現我們根據實際情況通過繼承ERC721BasicToken或者ERC721Token來添加自己的業務邏輯。
OpenZeppelin ERC721源碼分析到這里就結束了。
轉載請注明: 轉載自Ryan是菜鳥 | LNMP技術棧筆記
如果覺得本篇文章對您十分有益,何不 打賞一下
本文鏈接地址: OpenZeppelin ERC721源碼分析
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/24227.html
摘要:前提是擁有者必須要通過某些機制對這個請求進行確認,比如通過進行。事件,當被調用時,需要觸發該事件。允許從中轉出的數增加所有者允許花費代幣的數量。已經歸屬合約,其余歸還給所有者。計算已歸屬但尚未釋放的金額。源碼分析到這里就結束了。 ERC20:Ethereum Request for Comments 20,是一個基于以太坊代幣的接口標準(協議)。所有符合ERC-20標準的代幣都能立即兼...
摘要:從這節開始,我們將學習代幣標準以及加密收集資產等知識。聲明一個繼承的新合約,命名為。注意目前是一個草稿,還沒有正式商定的實現。所以把這一個可能的實現當作考慮,但不要把它作為代幣的官方標準。 從這節開始,我們將學習代幣, ERC721標準, 以及加密收集資產等知識。 一、代幣 代幣 讓我們來聊聊以太坊上的代幣。 如果你對以太坊的世界有一些了解,你很可能聽過人們聊到代幣——尤其是 ERC2...
摘要:合約安全增強溢出和下溢我們將來學習你在編寫智能合約的時候需要注意的一個主要的安全特性防止溢出和下溢。實戰演練給加上一些標簽把這里變成標準的注釋把一個管理轉移僵尸所有權的合約符合對標準草案的實現 通過上一節的學習,我們完成了 ERC721 的實現。并不是很復雜,對吧?很多類似的以太坊概念,當你只聽人們談論它們的時候,會覺得很復雜。所以最簡單的理解方式就是你自己來實現它。 一、預防溢出 不...
摘要:本文就來剖析下什么是是什么在創建代幣一篇,我們講到過代幣,和一樣,同樣是一個代幣標準,官方簡要解釋是,簡寫為,多翻譯為非同質代幣。返回合約代幣符號,盡管是可選,但強烈建議實現,即便是返回空字符串。 本文首發于深入淺出區塊鏈社區原文鏈接:剖析非同質化代幣ERC721-全面解析ERC721標準原文已更新,請讀者前往原文閱讀 什么是ERC-721?現在我們看到的各種加密貓貓狗狗都是基于ERC...
摘要:是企業與區塊鏈相遇的地方。的框架旨在成為開發區塊鏈解決方案的支柱。以太坊,主要是針對工程師使用進行區塊鏈以太坊開發的詳解。 如果你想將區塊鏈合并到一個Java項目中,現在我們來看看就是這個細分領域中三個最大的OSS玩家。 好的伙計們,我們都聽說過比特幣,以太坊或其他加密貨幣,其中有一些時髦的名字圍繞著我們常見的新聞,但我們作為Java開發人員知道如何輕松地與這些區塊鏈技術進行交互嗎?以...
閱讀 3822·2021-10-12 10:11
閱讀 3641·2021-09-13 10:27
閱讀 2546·2019-08-30 15:53
閱讀 1977·2019-08-29 18:33
閱讀 2197·2019-08-29 14:03
閱讀 1000·2019-08-29 13:27
閱讀 3322·2019-08-28 18:07
閱讀 774·2019-08-26 13:23