您好,登錄后才能下訂單哦!
這篇文章主要講解了“ERC721藏品合約怎么實現(xiàn)”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“ERC721藏品合約怎么實現(xiàn)”吧!
ERC721約定了一些接口函數(shù),使它在一定程度上符合ERC20代幣標(biāo)準(zhǔn)。這么做是為了讓現(xiàn)有的 錢包更容易顯示代幣的基本信息。這些函數(shù)可以讓符合ERC721標(biāo)準(zhǔn)的智能合約像比特幣或者 以太幣這樣普通的數(shù)字加密幣一樣,通過智能合約編程的方式定義一些功能讓用戶實現(xiàn)向他人 發(fā)送代幣或檢查賬戶余額等操作。
這是一個簡明的ERC721智能合約聲明:
contract ERC721 { //與ERC20兼容的接口 function name() constant returns (string name); function symbol() constant returns (string symbol); function totalSupply() constant returns (uint256 totalSupply); function balanceOf(address _owner) constant returns (uint balance); //所有權(quán)相關(guān)的接口 function ownerOf(uint256 _tokenId) constant returns (address owner); function approve(address _to, uint256 _tokenId); function takeOwnership(uint256 _tokenId); function transfer(address _to, uint256 _tokenId); function tokenOfOwnerByIndex(address _owner, uint256 _index) constant returns (uint tokenId); //通證元數(shù)據(jù)接口 function tokenMetadata(uint256 _tokenId) constant returns (string infoUrl); //事件 event Transfer(address indexed _from, address indexed _to, uint256 _tokenId); event Approval(address indexed _owner, address indexed _approved, uint256 _tokenId); }
該函數(shù)應(yīng)當(dāng)返回通證的名稱。 例如:
contract MyNFT { function name() constant returns(string name){ return "My Non-FungibleToken"; } }
該函數(shù)應(yīng)當(dāng)返回通證的符號,它有助于提高與ERC20的兼容性。例如:
contract MyNFT { function symbol() constant returns(string symbol){ return "MNFT"; } }
該函數(shù)應(yīng)當(dāng)返回區(qū)塊鏈上供應(yīng)的通證總數(shù)量,該數(shù)量不一定是固定不變的。 例如:
contract MyNFT { //想發(fā)行多少取決于你 ;) uint256 private totalSupply = 1000000000; function totalSupply() constant returns (uint256supply){ return totalSupply; } }
該函數(shù)用于查詢某一地址里的通證余額。例如:
contract MyNFT { mapping(address => uint) privatebalances; function balanceOf(address _owner) constant returns(uint balance){ return balances[_owner]; } }
下面這些函數(shù)定義了合約如何處理通證的所有權(quán)及如何轉(zhuǎn)移所有權(quán)。其中最重要的兩個函數(shù) 是獲取(takeOwnership)和轉(zhuǎn)賬(transfer),用來實現(xiàn)用戶之間的通證流轉(zhuǎn),就像銀行的提款 和匯款功能。
該函數(shù)返回通證持有人的地址。因為每一個ERC721通證都是不可替代的,因此可以在區(qū)塊鏈上 唯一的地址找到,我們可以用通證的ID來確定其持有人。
contract MyNFT { mapping(uint256 => address) privatetokenOwners; mapping(uint256 => bool) private tokenExists; function ownerOf(uint256 _tokenId) constant returns (address owner) { require(tokenExists[_tokenId]); return tokenOwners[_tokenId]; } }
該函數(shù)用來授權(quán)給另一主體代表持有人進(jìn)行通證轉(zhuǎn)移操作。例如,假設(shè)Alice有一個ERC721通證,她可以 調(diào)用approve
函數(shù)來授權(quán)給她的朋友Bob,然后Bob就可以代表Alice行使通證持有人的權(quán)利。
contract MyNFT { mapping(address => mapping (address=> uint256)) allowed; function approve(address _to, uint256 _tokenId){ require(msg.sender ==ownerOf(_tokenId)); require(msg.sender != _to); allowed[msg.sender][_to] = _tokenId; Approval(msg.sender, _to, _tokenId); } }
該函數(shù)類似于取款功能,一個外部主體通過調(diào)用takeOwnership
函數(shù)來從另一個用戶的賬戶 中提取ERC721通證。
因此,在一個用戶被(其他人)授權(quán)擁有一定數(shù)量的通證的情況下,可以通過該功能將這部分 通證從另一個用戶的賬戶中提取出來。
contract MyNFT { function takeOwnership(uint256_tokenId){ require(tokenExists[_tokenId]); address oldOwner = ownerOf(_tokenId); address newOwner = msg.sender; require(newOwner != oldOwner); require(allowed[oldOwner][newOwner] == _tokenId); balances[oldOwner] -= 1; tokenOwners[_tokenId] = newOwner; balances[oldOwner] += 1; Transfer(oldOwner, newOwner,_tokenId); } }
另一種轉(zhuǎn)移通證的方法時使用transfer
函數(shù)。轉(zhuǎn)賬(transfer)功能可以讓用戶將通證發(fā)給另一個用戶, 類似于操作比特幣這樣的加密數(shù)字貨幣。然而,只有在匯出賬戶之前授權(quán)過匯入賬戶持有其通證的 情況下,才可以進(jìn)行轉(zhuǎn)賬。
contract MyNFT { mapping(address => mapping(uint256 => uint256)) private ownerTokens; function removeFromTokenList(address owner, uint256 _tokenId) private { for(uint256 i = 0;ownerTokens[owner][i] != _tokenId;i++){ ownerTokens[owner][i] = 0; } } function transfer(address _to, uint256 _tokenId){ address currentOwner = msg.sender; address newOwner = _to; require(tokenExists[_tokenId]); require(currentOwner == ownerOf(_tokenId)); require(currentOwner != newOwner); require(newOwner != address(0)); removeFromTokenList(_tokenId); balances[oldOwner] -= 1; tokenOwners[_tokenId] = newOwner; balances[newOwner] += 1; Transfer(oldOwner, newOwner, _tokenId); } }
這個函數(shù)是可選的,但推薦你實現(xiàn)它。
每一個ERC721通證的持有者可以同時持有不止一個通證,因為每個通證都有唯一的ID,但是,要跟蹤某個用戶持有的 通證可能就會比較困難。為此,合約需要記錄每個用戶持有的每個通證。通過這種方式,用戶可以 通過索引清單檢索其擁有的通證。通證檢索(tokenOfOwnerByIndex)函數(shù)可以通過這種方式追溯某一特定的通證。
contract MyNFT { mapping(address => mapping(uint256 => uint256)) private ownerTokens; function tokenOfOwnerByIndex(address _owner, uint256 _index) constant returns (uint tokenId){ return ownerTokens[_owner][_index]; } }
就像我們之前所說的,使物品具有不可替代性的是它們獨一無二的特質(zhì)。美元和網(wǎng)球卡不可替代, 因為它們的特征不同。但是,在區(qū)塊鏈上將這些區(qū)分每個通證的特征儲存下來成本很高,也不推薦這么做。 為了解決這個問題,我們可以儲存每個通證的引用(references),例如IPFS哈?;騂TTP(S)鏈接,這些 引用,被稱作元數(shù)據(jù)。元數(shù)據(jù)是可選的。
tokenMetaData函數(shù)應(yīng)當(dāng)返回通證的元數(shù)據(jù),或者通證數(shù)據(jù)的鏈接。
contract MyNFT { mapping(uint256 => string) tokenLinks; function tokenMetadata(uint256 _tokenId) constant returns (string infoUrl) { return tokenLinks[_tokenId]; } }
當(dāng)調(diào)用合約方法的時候,事件將會被觸發(fā),并且一旦被觸發(fā)就會向監(jiān)聽系統(tǒng)傳播。外部應(yīng)用可以監(jiān)聽區(qū)塊鏈 中的事件,一旦接收到區(qū)塊鏈中的事件被觸發(fā),監(jiān)聽系統(tǒng)就可以通過事件中包含的信息執(zhí)行邏輯程序。 ERC721標(biāo)準(zhǔn)定義了下面兩個事件。
當(dāng)一個通證的所有權(quán)從一個用戶轉(zhuǎn)移到另一個時,將觸發(fā)該事件,事件的信息包括匯出賬戶、匯入賬戶和通證ID。
contract MyNFT { event Transfer(address indexed _from,address indexed _to, uint256 _tokenId); }
當(dāng)一個用戶允許另一個用戶持有其通證的時候(例如啟用“授權(quán)”功能的時候),該事件就會被觸發(fā),事件的信息中 包含這些通證現(xiàn)在的持有賬戶、被授權(quán)賬戶以及通證ID。
contract MyNFT { event Approval(address indexed _owner, address indexed _approved, uint256 _tokenId); }
感謝各位的閱讀,以上就是“ERC721藏品合約怎么實現(xiàn)”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對ERC721藏品合約怎么實現(xiàn)這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。