ERC-20 कॉन्ट्रैक्ट वॉक-थ्रू
परिचय
एथेरियम के लिए सबसे आम उपयोगों में से एक समूह के लिए एक व्यापार योग्य टोकन बनाना है, एक अर्थ में उनकी अपनी मुद्रा। ये टोकन आमतौर पर एक मानक का पालन करते हैं, ERC-20। यह मानक लिक्विडिटी पूल और वॉलेट जैसे उपकरण लिखना संभव बनाता है, जो सभी ERC-20 टोकन के साथ काम करते हैं। इस लेख में हम ओपनज़ेपेलिन सॉलिडिटी ERC20 इम्प्लीमेंटेशन (opens in a new tab), और इंटरफ़ेस परिभाषा (opens in a new tab) का विश्लेषण करेंगे।
यह एनोटेटेड सोर्स कोड है। अगर आप ERC-20 लागू करना चाहते हैं, तो यह ट्यूटोरियल पढ़ें (opens in a new tab)।
इंटरफ़ेस
ERC-20 जैसे मानक का उद्देश्य कई टोकन कार्यान्वयनों को अनुमति देना है जो वॉलेट और विकेन्द्रीकृत एक्सचेंजों जैसे अनुप्रयोगों में इंटरऑपरेबल हैं। इसे प्राप्त करने के लिए, हम एक इंटरफ़ेस (opens in a new tab) बनाते हैं। कोई भी कोड जिसे टोकन कॉन्ट्रैक्ट का उपयोग करने की आवश्यकता है, वह इंटरफ़ेस में समान परिभाषाओं का उपयोग कर सकता है और इसका उपयोग करने वाले सभी टोकन कॉन्ट्रैक्ट के साथ संगत हो सकता है, चाहे वह मेटामास्क जैसा वॉलेट हो, etherscan.io जैसा डैप हो, या लिक्विडिटी पूल जैसा कोई अलग कॉन्ट्रैक्ट हो।
यदि आप एक अनुभवी प्रोग्रामर हैं, तो आपको शायद जावा (opens in a new tab) या यहां तक कि C हेडर फ़ाइलों (opens in a new tab) में भी समान संरचनाएं देखने को याद होंगी।
यह ओपनज़ेपेलिन से ERC-20 इंटरफेस (opens in a new tab) की एक परिभाषा है। यह मानव पठनीय मानक (opens in a new tab) का सॉलिडिटी कोड में अनुवाद है। बेशक, इंटरफ़ेस स्वयं यह परिभाषित नहीं करता है कि कुछ कैसे करना है। यह नीचे दिए गए कॉन्ट्रैक्ट सोर्स कोड में समझाया गया है।
1// SPDX-License-Identifier: MITसॉलिडिटी फ़ाइलों में एक लाइसेंस पहचानकर्ता शामिल होना चाहिए। आप यहां लाइसेंसों की सूची देख सकते हैं (opens in a new tab)। यदि आपको एक अलग लाइसेंस की आवश्यकता है, तो बस इसे कमेंट में समझाएं।
1pragma solidity >=0.6.0 <0.8.0;सॉलिडिटी भाषा अभी भी तेजी से विकसित हो रही है, और नए संस्करण पुराने कोड के साथ संगत नहीं हो सकते हैं (यहां देखें (opens in a new tab))। इसलिए, यह एक अच्छा विचार है कि न केवल भाषा का न्यूनतम संस्करण निर्दिष्ट किया जाए, बल्कि एक अधिकतम संस्करण भी, नवीनतम जिसके साथ आपने कोड का परीक्षण किया है।
1/**2 * @dev EIP में परिभाषित ERC20 मानक का इंटरफ़ेस।3 */कमेंट में @dev नैटस्पेक फॉर्मेट (opens in a new tab) का हिस्सा है, जिसका उपयोग सोर्स कोड से
प्रलेखन बनाने के लिए किया जाता है।
1interface IERC20 {परंपरा के अनुसार, इंटरफ़ेस के नाम I से शुरू होते हैं।
1 /**2 * @dev अस्तित्व में टोकन की मात्रा लौटाता है।3 */4 function totalSupply() external view returns (uint256);यह फ़ंक्शन external है, जिसका अर्थ है इसे केवल अनुबंध के बाहर से कॉल किया जा सकता है (opens in a new tab)।
यह कॉन्ट्रैक्ट में टोकन की कुल आपूर्ति लौटाता है। यह मान एथेरियम में सबसे सामान्य प्रकार, अनसाईंड 256 बिट्स का उपयोग करके लौटाया जाता है (256 बिट EVM का
नेटिव शब्द आकार है)। यह फ़ंक्शन एक view भी है, जिसका अर्थ है कि यह स्टेट को नहीं बदलता है, इसलिए इसे ब्लॉकचेन के प्रत्येक नोड पर चलाने के बजाय
एकल नोड पर निष्पादित किया जा सकता है। इस तरह का फ़ंक्शन लेनदेन उत्पन्न नहीं करता है और इसमें गैस नहीं लगती है।
ध्यान दें: सिद्धांत रूप में ऐसा लग सकता है कि किसी कॉन्ट्रैक्ट का निर्माता वास्तविक मूल्य से कम कुल आपूर्ति लौटाकर धोखा दे सकता है, जिससे प्रत्येक टोकन वास्तविक से अधिक मूल्यवान प्रतीत होता है। हालांकि, यह डर ब्लॉकचेन की वास्तविक प्रकृति को नजरअंदाज करता है। ब्लॉकचेन पर होने वाली हर चीज़ को हर नोड द्वारा सत्यापित किया जा सकता है। इसे प्राप्त करने के लिए, प्रत्येक कॉन्ट्रैक्ट का मशीन भाषा कोड और स्टोरेज हर नोड पर उपलब्ध है। हालांकि आपको अपने अनुबंध के लिए सॉलिडिटी कोड प्रकाशित करने की आवश्यकता नहीं है, लेकिन कोई भी आपको गंभीरता से नहीं लेगा जब तक कि आप स्रोत कोड और सॉलिडिटी का वह संस्करण प्रकाशित नहीं करते जिसके साथ इसे संकलित किया गया था, ताकि इसे आपके द्वारा प्रदान किए गए मशीन भाषा कोड के विरुद्ध सत्यापित किया जा सके। उदाहरण के लिए, यह अनुबंध देखें (opens in a new tab)।
1 /**2 * @dev `खाते` के स्वामित्व वाले टोकन की राशि लौटाता है।3 */4 function balanceOf(address account) external view returns (uint256);जैसा कि नाम से पता चलता है, balanceOf एक खाते का बैलेंस लौटाता है। एथेरियम खातों को address प्रकार का उपयोग करके सॉलिडिटी में पहचाना जाता है, जो 160 बिट्स रखता है।
यह external और view भी है।
1 /**2 * @dev कॉलर के खाते से `recipient` को `राशि` टोकन ले जाता है।3 *4 * एक बूलियन मान लौटाता है जो यह दर्शाता है कि ऑपरेशन सफल हुआ या नहीं।5 *6 * {ट्रांसफर} इवेंट उत्सर्जित करता है।7 */8 function transfer(address recipient, uint256 amount) external returns (bool);transfer फ़ंक्शन कॉलर से एक अलग पते पर टोकन स्थानांतरित करता है। इसमें स्टेट में बदलाव शामिल है, इसलिए यह view नहीं है।
जब कोई यूज़र इस फ़ंक्शन को कॉल करता है तो यह एक लेनदेन बनाता है और इसमें गैस लगती है। यह एक इवेंट, Transfer भी उत्सर्जित करता है, ताकि
ब्लॉकचेन पर सभी को इवेंट की सूचना दी जा सके।
फ़ंक्शन में दो अलग-अलग प्रकार के कॉलर्स के लिए दो प्रकार के आउटपुट होते हैं:
- उपयोगकर्ता जो सीधे उपयोगकर्ता इंटरफ़ेस से फ़ंक्शन को कॉल करते हैं। आमतौर पर उपयोगकर्ता एक लेनदेन सबमिट करता है
और प्रतिक्रिया की प्रतीक्षा नहीं करता है, जिसमें अनिश्चित काल तक का समय लग सकता है। उपयोगकर्ता यह देख सकता है कि क्या हुआ
लेनदेन रसीद (जिसे लेनदेन हैश द्वारा पहचाना जाता है) को देखकर या
ट्रांसफरइवेंट को देखकर। - अन्य अनुबंध, जो समग्र लेनदेन के हिस्से के रूप में फ़ंक्शन को कॉल करते हैं। उन अनुबंधों को तुरंत परिणाम मिलता है, क्योंकि वे एक ही लेनदेन में चलते हैं, इसलिए वे फ़ंक्शन वापसी मान का उपयोग कर सकते हैं।
अनुबंध की स्थिति को बदलने वाले अन्य कार्यों द्वारा समान प्रकार का आउटपुट बनाया जाता है।
अलाउंस एक खाते को कुछ ऐसे टोकन खर्च करने की अनुमति देते हैं जो एक अलग मालिक के हैं। यह उपयोगी है, उदाहरण के लिए, उन अनुबंधों के लिए जो विक्रेताओं के रूप में कार्य करते हैं। अनुबंध इवेंट के लिए मॉनिटर नहीं कर सकते हैं, इसलिए यदि कोई खरीदार सीधे विक्रेता अनुबंध को टोकन स्थानांतरित करता है तो उस अनुबंध को यह नहीं पता होगा कि इसका भुगतान किया गया था। इसके बजाय, खरीदार विक्रेता अनुबंध को एक निश्चित राशि खर्च करने की अनुमति देता है, और विक्रेता उस राशि को स्थानांतरित करता है। यह एक फ़ंक्शन के माध्यम से किया जाता है जिसे विक्रेता अनुबंध कॉल करता है, इसलिए विक्रेता अनुबंध जान सकता है कि यह सफल था या नहीं।
1 /**2 * @dev टोकन की शेष संख्या लौटाता है जिसे `spender` को3 * {transferFrom} के माध्यम से `owner` की ओर से खर्च करने की अनुमति दी जाएगी। यह4 * डिफ़ॉल्ट रूप से शून्य है।5 *6 * जब {approve} या {transferFrom} को कॉल किया जाता है तो यह मान बदल जाता है।7 */8 function allowance(address owner, address spender) external view returns (uint256);allowance फ़ंक्शन किसी को भी यह देखने के लिए क्वेरी करने देता है कि एक
पता (owner) दूसरे पते (spender) को कितना खर्च करने देता है।
1 /**2 * @dev कॉलर के टोकन पर `spender` के भत्ते के रूप में `राशि` सेट करता है।3 *4 * एक बूलियन मान लौटाता है जो यह दर्शाता है कि ऑपरेशन सफल हुआ या नहीं।5 *6 * महत्वपूर्ण: सावधान रहें कि इस पद्धति के साथ भत्ता बदलने से यह जोखिम होता है7 * कि कोई दुर्भाग्यपूर्ण लेनदेन8 * क्रम से पुराने और नए दोनों भत्तों का उपयोग कर सकता है। इस रेस कंडीशन को9 * कम करने का एक संभावित समाधान यह है कि पहले खर्च करने वाले के भत्ते को 0 तक कम किया जाए और बाद में10 * वांछित मान सेट किया जाए:11 * https://github.com/ethereum/EIPs/issues/20#issuecomment-26352472912 *13 * एक {Approval} इवेंट उत्सर्जित करता है।14 */15 function approve(address spender, uint256 amount) external returns (bool);सभी दिखाएँapprove फ़ंक्शन एक अलाउंस बनाता है। इसके
दुरुपयोग के बारे में संदेश पढ़ना सुनिश्चित करें। एथेरियम में आप अपने स्वयं के लेनदेन के क्रम को नियंत्रित करते हैं,
लेकिन आप उस क्रम को नियंत्रित नहीं कर सकते जिसमें अन्य लोगों के लेनदेन
निष्पादित होंगे, जब तक कि आप अपना स्वयं का लेनदेन तब तक सबमिट नहीं करते जब तक आप यह नहीं देखते कि
दूसरे पक्ष का लेनदेन हो गया है।
1 /**2 * @dev `सेंडर` से `प्राप्तकर्ता` तक `राशि` टोकन को3 * भत्ता तंत्र का उपयोग करके ले जाता है। `राशि` को फिर कॉलर के4 * भत्ते से काट लिया जाता है।5 *6 * एक बूलियन मान लौटाता है जो यह दर्शाता है कि ऑपरेशन सफल हुआ या नहीं।7 *8 * एक {ट्रांसफर} इवेंट उत्सर्जित करता है।9 */10 function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);सभी दिखाएँअंत में, transferFrom का उपयोग खर्च करने वाले द्वारा वास्तव में भत्ता खर्च करने के लिए किया जाता है।
12 /**3 * @dev जब `मान` टोकन एक खाते (`from`) से4 * दूसरे (`to`) में ले जाए जाते हैं तो उत्सर्जित होता है।5 *6 * ध्यान दें कि `मान` शून्य हो सकता है।7 */8 event Transfer(address indexed from, address indexed to, uint256 value);910 /**11 * @dev जब {approve} पर कॉल द्वारा `owner` के लिए `spender` का भत्ता सेट किया जाता है12 * तो उत्सर्जित होता है। `value` नया भत्ता है।13 */14 event Approval(address indexed owner, address indexed spender, uint256 value);15}सभी दिखाएँये इवेंट तब उत्सर्जित होते हैं जब ERC-20 अनुबंध की स्थिति बदल जाती है।
वास्तविक अनुबंध
यह वास्तविक अनुबंध है जो ERC-20 मानक को लागू करता है, यहां से लिया गया (opens in a new tab)। इसका उपयोग जैसा है वैसा करने के लिए नहीं है, लेकिन आप इसे कुछ प्रयोग करने योग्य बनाने के लिए इससे इनहेरिट (opens in a new tab) कर सकते हैं।
1// SPDX-License-Identifier: MIT2pragma solidity >=0.6.0 <0.8.0;
इंपोर्ट स्टेटमेंट
ऊपर दी गई इंटरफ़ेस परिभाषाओं के अलावा, अनुबंध परिभाषा दो अन्य फ़ाइलों को आयात करती है:
12import "../../GSN/Context.sol";3import "./IERC20.sol";4import "../../math/SafeMath.sol";GSN/Context.solओपनजीएसएन (opens in a new tab) का उपयोग करने के लिए आवश्यक परिभाषाएं हैं, एक ऐसी प्रणाली जो ईथर के बिना उपयोगकर्ताओं को ब्लॉकचेन का उपयोग करने की अनुमति देती है। ध्यान दें कि यह एक पुराना संस्करण है, यदि आप ओपनजीएसएन के साथ एकीकृत करना चाहते हैं इस ट्यूटोरियल का उपयोग करें (opens in a new tab)।- SafeMath लाइब्रेरी (opens in a new tab), जो सॉलिडिटी संस्करणों <0.8.0 के लिए अरिथमैटिक ओवरफ़्लो/अंडरफ़्लो को रोकती है। सॉलिडिटी ≥0.8.0 में, अंकगणितीय संचालन स्वचालित रूप से ओवरफ़्लो/अंडरफ़्लो पर वापस आ जाते हैं, जिससे SafeMath अनावश्यक हो जाता है। यह अनुबंध पुराने संकलक संस्करणों के साथ पश्चगामी संगतता के लिए SafeMath का उपयोग करता है।
यह टिप्पणी अनुबंध के उद्देश्य को बताती है।
1/**2 * @dev {IERC20} इंटरफ़ेस का कार्यान्वयन।3 *4 * यह कार्यान्वयन उस तरीके से अज्ञेयवादी है जिस तरह से टोकन बनाए जाते हैं। इसका मतलब है5 * कि {_mint} का उपयोग करके एक व्युत्पन्न अनुबंध में एक आपूर्ति तंत्र जोड़ा जाना है।6 * एक सामान्य तंत्र के लिए {ERC20PresetMinterPauser} देखें।7 *8 * टिप: विस्तृत विवरण के लिए हमारी मार्गदर्शिका देखें9 * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[कैसे10 * आपूर्ति तंत्र लागू करें]।11 *12 * हमने सामान्य OpenZeppelin दिशानिर्देशों का पालन किया है: फ़ंक्शन विफलता पर13 * `गलत` लौटाने के बजाय वापस आ जाते हैं। यह व्यवहार फिर भी पारंपरिक है14 * और ERC20 अनुप्रयोगों की अपेक्षाओं के साथ टकराव नहीं करता है।15 *16 * इसके अतिरिक्त, {transferFrom} पर कॉल पर एक {Approval} इवेंट उत्सर्जित होता है।17 * यह अनुप्रयोगों को केवल18 * उक्त घटनाओं को सुनकर सभी खातों के लिए भत्ता फिर से बनाने की अनुमति देता है। EIP के अन्य कार्यान्वयन19 * इन घटनाओं का उत्सर्जन नहीं कर सकते हैं, क्योंकि यह विनिर्देश द्वारा आवश्यक नहीं है।20 *21 * अंत में, गैर-मानक {decreaseAllowance} और {increaseAllowance}22 * फ़ंक्शंस को भत्ता23 * सेट करने के आसपास के प्रसिद्ध मुद्दों को कम करने के लिए जोड़ा गया है। {IERC20-approve} देखें।24 */25सभी दिखाएँअनुबंध परिभाषा
1contract ERC20 is Context, IERC20 {यह पंक्ति इनहेरिटेंस को निर्दिष्ट करती है, इस मामले में ऊपर से IERC20 और Context से, ओपनजीएसएन के लिए।
12 using SafeMath for uint256;3यह पंक्ति SafeMath लाइब्रेरी को uint256 प्रकार से जोड़ती है। आप इस लाइब्रेरी को
यहां (opens in a new tab) पा सकते हैं।
वैरिएबल परिभाषाएं
ये परिभाषाएँ अनुबंध के स्टेट वैरिएबल को निर्दिष्ट करती हैं। इन वैरिएबल को private घोषित किया गया है, लेकिन
इसका केवल यह मतलब है कि ब्लॉकचेन पर अन्य अनुबंध उन्हें नहीं पढ़ सकते हैं। ब्लॉकचेन पर कोई
राज नहीं है, हर नोड पर सॉफ्टवेयर में हर ब्लॉक पर हर अनुबंध की स्टेट होती है। परंपरा के अनुसार, स्टेट वैरिएबल का नाम _<something> होता है।
पहले दो वैरिएबल मैपिंग (opens in a new tab) हैं, मतलब वे मोटे तौर पर एसोसिएटिव एरे (opens in a new tab) के समान व्यवहार करते हैं, सिवाय इसके कि कुंजी संख्यात्मक मान हैं। भंडारण केवल उन प्रविष्टियों के लिए आवंटित किया जाता है जिनके मान डिफ़ॉल्ट (शून्य) से अलग हैं।
1 mapping (address => uint256) private _balances;पहली मैपिंग, _balances, इस टोकन के पते और उनके संबंधित बैलेंस हैं। बैलेंस तक पहुँचने के लिए,
इस सिंटैक्स का उपयोग करें: _balances[<address>]।
1 mapping (address => mapping (address => uint256)) private _allowances;यह वैरिएबल, _allowances, पहले बताए गए अलाउंस को स्टोर करता है। पहला इंडेक्स टोकन का
मालिक है, और दूसरा अलाउंस के साथ अनुबंध है। पता A पते B के खाते से
कितनी राशि खर्च कर सकता है, यह जानने के लिए, _allowances[B][A] का उपयोग करें।
1 uint256 private _totalSupply;जैसा कि नाम से पता चलता है, यह वैरिएबल टोकन की कुल आपूर्ति का ट्रैक रखता है।
1 string private _name;2 string private _symbol;3 uint8 private _decimals;इन तीन वैरिएबल का उपयोग पठनीयता में सुधार के लिए किया जाता है। पहले दो स्व-व्याख्यात्मक हैं, लेकिन _decimals
नहीं है।
एक तरफ, एथेरियम में फ्लोटिंग पॉइंट या भिन्नात्मक वैरिएबल नहीं हैं। दूसरी ओर, मनुष्य टोकन को विभाजित करने में सक्षम होना पसंद करते हैं। लोगों द्वारा मुद्रा के लिए सोने को अपनाने का एक कारण यह था कि जब कोई गाय के मूल्य की बत्तख खरीदना चाहता था तो छुट्टे बनाना मुश्किल था।
समाधान पूर्णांकों का ट्रैक रखना है, लेकिन वास्तविक टोकन के बजाय एक भिन्नात्मक टोकन की गणना करना है जो लगभग बेकार है। ईथर के मामले में, भिन्नात्मक टोकन को wei कहा जाता है, और 10^18 wei एक ETH के बराबर होता है। लिखते समय, 10,000,000,000,000 wei लगभग एक अमेरिकी या यूरो सेंट के बराबर है।
एप्लिकेशन को यह जानना होगा कि टोकन बैलेंस कैसे प्रदर्शित करें। यदि किसी उपयोगकर्ता के पास 3,141,000,000,000,000,000 wei है, तो क्या वह
3.14 ETH है? 31.41 ETH? 3,141 ETH? ईथर के मामले में इसे ETH के लिए 10^18 wei परिभाषित किया गया है, लेकिन अपने
टोकन के लिए आप एक अलग मान चुन सकते हैं। यदि टोकन को विभाजित करने का कोई मतलब नहीं है, तो आप शून्य के _decimals मान का उपयोग कर सकते हैं। यदि आप ETH के समान मानक का उपयोग करना चाहते हैं, तो मान 18 का उपयोग करें।
कंस्ट्रक्टर
1 /**2 * @dev {नाम} और {प्रतीक} के लिए मान सेट करता है, {दशमलव} को3 * 18 के डिफ़ॉल्ट मान के साथ प्रारंभ करता है।4 *5 * {दशमलव} के लिए एक अलग मान का चयन करने के लिए, {_setupDecimals} का उपयोग करें।6 *7 * ये तीनों मान अपरिवर्तनीय हैं: इन्हें8 * निर्माण के दौरान केवल एक बार सेट किया जा सकता है।9 */10 constructor (string memory name_, string memory symbol_) public {11 // सॉलिडिटी ≥0.7.0 में, 'पब्लिक' अंतर्निहित है और इसे छोड़ा जा सकता है।1213 _name = name_;14 _symbol = symbol_;15 _decimals = 18;16 }सभी दिखाएँकंस्ट्रक्टर को तब कॉल किया जाता है जब अनुबंध पहली बार बनाया जाता है। परंपरा के अनुसार, फ़ंक्शन पैरामीटर का नाम <something>_ होता है।
यूज़र इंटरफ़ेस फ़ंक्शंस
1 /**2 * @dev टोकन का नाम लौटाता है।3 */4 function name() public view returns (string memory) {5 return _name;6 }78 /**9 * @dev टोकन का प्रतीक लौटाता है, जो आमतौर पर नाम का एक10 * छोटा संस्करण होता है।11 */12 function symbol() public view returns (string memory) {13 return _symbol;14 }1516 /**17 * @dev इसका उपयोगकर्ता प्रतिनिधित्व प्राप्त करने के लिए उपयोग किए जाने वाले दशमलव की संख्या लौटाता है।18 * उदाहरण के लिए, यदि `दशमलव` `2` के बराबर है, तो `505` टोकन का बैलेंस19 * उपयोगकर्ता को `5,05` (`505 / 10 ** 2`) के रूप में प्रदर्शित किया जाना चाहिए।20 *21 * टोकन आमतौर पर 18 के मान का विकल्प चुनते हैं, जो22 * ईथर और वी के बीच के संबंध की नकल करता है। यह वह मान है जिसका उपयोग {ERC20} करता है, जब तक कि {_setupDecimals} को23 * कॉल न किया जाए।24 *25 * ध्यान दें: यह जानकारी केवल _प्रदर्शन_ उद्देश्यों के लिए उपयोग की जाती है: यह26 * अनुबंध के किसी भी अंकगणित को किसी भी तरह से प्रभावित नहीं करती है, जिसमें27 * {IERC20-बैलेंसऑफ} और {IERC20-ट्रांसफर} शामिल हैं।28 */29 function decimals() public view returns (uint8) {30 return _decimals;31 }सभी दिखाएँये फ़ंक्शन, name, symbol, और decimals उपयोगकर्ता इंटरफ़ेस को आपके अनुबंध के बारे में जानने में मदद करते हैं ताकि वे इसे ठीक से प्रदर्शित कर सकें।
रिटर्न प्रकार string memory है, जिसका अर्थ है मेमोरी में संग्रहीत एक स्ट्रिंग लौटाना। वैरिएबल, जैसे
स्ट्रिंग्स, तीन स्थानों पर संग्रहीत किए जा सकते हैं:
| जीवनकाल | अनुबंध एक्सेस | गैस लागत | |
|---|---|---|---|
| मेमोरी | फ़ंक्शन कॉल | पढ़ें/लिखें | दसियों या सैकड़ों (उच्च स्थानों के लिए उच्च) |
| Calldata | फ़ंक्शन कॉल | केवल पढ़ें | रिटर्न प्रकार के रूप में उपयोग नहीं किया जा सकता है, केवल एक फ़ंक्शन पैरामीटर प्रकार |
| स्टोरेज | बदलने तक | पढ़ें/लिखें | उच्च (पढ़ने के लिए 800, लिखने के लिए 20k) |
इस मामले में, memory सबसे अच्छा विकल्प है।
टोकन जानकारी पढ़ें
ये फ़ंक्शन हैं जो टोकन के बारे में जानकारी प्रदान करते हैं, या तो कुल आपूर्ति या किसी खाते का बैलेंस।
1 /**2 * @dev {IERC20-totalSupply} देखें।3 */4 function totalSupply() public view override returns (uint256) {5 return _totalSupply;6 }totalSupply फ़ंक्शन टोकन की कुल आपूर्ति लौटाता है।
1 /**2 * @dev {IERC20-balanceOf} देखें।3 */4 function balanceOf(address account) public view override returns (uint256) {5 return _balances[account];6 }किसी खाते का बैलेंस पढ़ें। ध्यान दें कि किसी को भी किसी और के खाते का बैलेंस प्राप्त करने की अनुमति है। इस जानकारी को छिपाने की कोशिश करने का कोई मतलब नहीं है, क्योंकि यह वैसे भी हर नोड पर उपलब्ध है। ब्लॉकचेन पर कोई रहस्य नहीं है।
टोकन स्थानांतरित करें
1 /**2 * @dev {IERC20-ट्रांसफर} देखें।3 *4 * आवश्यकताएं:5 *6 * - `प्राप्तकर्ता` शून्य पता नहीं हो सकता है।7 * - कॉलर के पास कम से कम `राशि` का बैलेंस होना चाहिए।8 */9 function transfer(address recipient, uint256 amount) public virtual override returns (bool) {सभी दिखाएँtransfer फ़ंक्शन को प्रेषक के खाते से किसी भिन्न खाते में टोकन स्थानांतरित करने के लिए कहा जाता है। ध्यान दें
कि भले ही यह एक बूलियन मान लौटाता है, वह मान हमेशा सत्य होता है। यदि ट्रांसफर
विफल हो जाता है तो अनुबंध कॉल को वापस कर देता है।
1 _transfer(_msgSender(), recipient, amount);2 return true;3 }_transfer फ़ंक्शन वास्तविक कार्य करता है। यह एक निजी फ़ंक्शन है जिसे केवल
अन्य अनुबंध कार्यों द्वारा ही कॉल किया जा सकता है। परंपरा के अनुसार निजी कार्यों का नाम _<something> होता है, जो स्टेट
वैरिएबल के समान है।
आम तौर पर सॉलिडिटी में हम संदेश भेजने वाले के लिए msg.sender का उपयोग करते हैं। हालांकि, यह
ओपनजीएसएन (opens in a new tab) को तोड़ता है। यदि हम अपने टोकन के साथ ईथरलेस लेनदेन की अनुमति देना चाहते हैं, तो हमें
_msgSender() का उपयोग करने की आवश्यकता है। यह सामान्य लेनदेन के लिए msg.sender लौटाता है, लेकिन ईथरलेस के लिए
मूल हस्ताक्षरकर्ता लौटाता है न कि उस अनुबंध को जिसने संदेश रिले किया था।
अलाउंस फ़ंक्शंस
ये फ़ंक्शन हैं जो अलाउंस की कार्यक्षमता को लागू करते हैं: allowance, approve, transferFrom,
और _approve। इसके अतिरिक्त, ओपनज़ेपेलिन कार्यान्वयन सुरक्षा में सुधार करने वाली कुछ सुविधाओं को शामिल करने के लिए मूल मानक से परे जाता है:
increaseAllowance, और decreaseAllowance।
अलाउंस फ़ंक्शन
1 /**2 * @dev {IERC20-अलाउंस} देखें।3 */4 function allowance(address owner, address spender) public view virtual override returns (uint256) {5 return _allowances[owner][spender];6 }allowance फ़ंक्शन सभी को किसी भी अलाउंस की जांच करने की अनुमति देता है।
अप्रूव फ़ंक्शन
1 /**2 * @dev {IERC20-अनुमोदन} देखें।3 *4 * आवश्यकताएं:5 *6 * - `खर्च करने वाला` शून्य पता नहीं हो सकता।7 */8 function approve(address spender, uint256 amount) public virtual override returns (bool) {इस फ़ंक्शन को अलाउंस बनाने के लिए कहा जाता है। यह ऊपर दिए गए transfer फ़ंक्शन के समान है:
- फ़ंक्शन बस एक आंतरिक फ़ंक्शन (इस मामले में,
_approve) को कॉल करता है जो वास्तविक कार्य करता है। - फ़ंक्शन या तो
true(यदि सफल हो) लौटाता है या रिवर्ट करता है (यदि नहीं)।
1 _approve(_msgSender(), spender, amount);2 return true;3 }हम उन जगहों की संख्या को कम करने के लिए आंतरिक कार्यों का उपयोग करते हैं जहां स्टेट परिवर्तन होते हैं। कोई भी फ़ंक्शन जो स्टेट को बदलता है, एक संभावित सुरक्षा जोखिम है जिसकी सुरक्षा के लिए ऑडिट करने की आवश्यकता है। इस तरह हमारे गलत होने की संभावना कम होती है।
transferFrom फ़ंक्शन
यह वह फ़ंक्शन है जिसे एक स्पेंडर अलाउंस खर्च करने के लिए कॉल करता है। इसके लिए दो संचालन की आवश्यकता होती है: खर्च की जा रही राशि को स्थानांतरित करें और उस राशि से अलाउंस कम करें।
1 /**2 * @dev {IERC20-transferFrom} देखें।3 *4 * अद्यतनित भत्ते को इंगित करने वाला एक {अनुमोदन} इवेंट उत्सर्जित करता है। यह5 * EIP द्वारा आवश्यक नहीं है। {ERC20} की शुरुआत में नोट देखें।6 *7 * आवश्यकताएं:8 *9 * - `प्रेषक` और `प्राप्तकर्ता` शून्य पता नहीं हो सकते।10 * - `प्रेषक` के पास कम से कम `राशि` का बैलेंस होना चाहिए।11 * - कॉलर के पास कम से कम `राशि` के ``प्रेषक`` के टोकन के लिए भत्ता होना चाहिए।12 */13 function transferFrom(address sender, address recipient, uint256 amount) public virtual14 override returns (bool) {15 _transfer(sender, recipient, amount);सभी दिखाएँ
a.sub(b, "संदेश") फ़ंक्शन कॉल दो काम करता है। सबसे पहले, यह a-b की गणना करता है, जो नया अलाउंस है।
दूसरा, यह जांचता है कि यह परिणाम नकारात्मक नहीं है। यदि यह नकारात्मक है तो कॉल दिए गए संदेश के साथ रिवर्ट हो जाता है। ध्यान दें कि जब कोई कॉल रिवर्ट होता है तो उस कॉल के दौरान पहले की गई कोई भी प्रोसेसिंग अनदेखा कर दी जाती है इसलिए हमें _transfer को
पूर्ववत करने की आवश्यकता नहीं है।
1 _approve(sender, _msgSender(), _allowances[sender][_msgSender()].sub(amount,2 "ERC20: transfer amount exceeds allowance"));3 return true;4 }ओपनज़ेपेलिन सुरक्षा परिवर्धन
गैर-शून्य भत्ते को किसी अन्य गैर-शून्य मान पर सेट करना खतरनाक है, क्योंकि आप केवल अपने स्वयं के लेनदेन के क्रम को नियंत्रित करते हैं, किसी और के नहीं। कल्पना कीजिए कि आपके पास दो उपयोगकर्ता हैं, ऐलिस जो भोली है और बिल जो बेईमान है। ऐलिस बिल से कुछ सेवा चाहती है, जिसकी उसे लगता है कि कीमत पांच टोकन है - इसलिए वह बिल को पांच टोकन का भत्ता देती है।
फिर कुछ बदलता है और बिल की कीमत दस टोकन तक बढ़ जाती है। ऐलिस, जो अभी भी सेवा चाहती है, एक लेनदेन भेजती है जो बिल के भत्ते को दस पर सेट करती है। जिस क्षण बिल इस नए लेनदेन को लेनदेन पूल में देखता है, वह एक लेनदेन भेजता है जो ऐलिस के पांच टोकन खर्च करता है और इसकी गैस कीमत बहुत अधिक होती है ताकि इसे तेजी से माइन किया जा सके। इस तरह बिल पहले पांच टोकन खर्च कर सकता है और फिर, एक बार जब ऐलिस का नया भत्ता माइन हो जाता है, तो पंद्रह टोकन की कुल कीमत के लिए दस और खर्च कर सकता है, जो ऐलिस द्वारा अधिकृत करने का इरादा था। इस तकनीक को फ्रंट-रनिंग (opens in a new tab) कहा जाता है
| ऐलिस लेनदेन | ऐलिस नॉन्स | बिल लेनदेन | बिल नॉन्स | बिल का भत्ता | ऐलिस से बिल की कुल आय |
|---|---|---|---|---|---|
| approve(Bill, 5) | 10 | 5 | 0 | ||
| transferFrom(Alice, Bill, 5) | 10,123 | 0 | 5 | ||
| approve(Bill, 10) | 11 | 10 | 5 | ||
| transferFrom(Alice, Bill, 10) | 10,124 | 0 | 15 |
इस समस्या से बचने के लिए, ये दो फ़ंक्शन (increaseAllowance और decreaseAllowance) आपको
एक विशिष्ट राशि से भत्ते को संशोधित करने की अनुमति देते हैं। तो अगर बिल ने पहले ही पांच टोकन खर्च कर दिए थे, तो वह सिर्फ
पांच और खर्च कर पाएगा। समय के आधार पर, यह दो तरीकों से काम कर सकता है, दोनों में
बिल को केवल दस टोकन मिलते हैं:
A:
| ऐलिस लेनदेन | ऐलिस नॉन्स | बिल लेनदेन | बिल नॉन्स | बिल का भत्ता | ऐलिस से बिल की कुल आय |
|---|---|---|---|---|---|
| approve(Bill, 5) | 10 | 5 | 0 | ||
| transferFrom(Alice, Bill, 5) | 10,123 | 0 | 5 | ||
| increaseAllowance(Bill, 5) | 11 | 0+5 = 5 | 5 | ||
| transferFrom(Alice, Bill, 5) | 10,124 | 0 | 10 |
B:
| ऐलिस लेनदेन | ऐलिस नॉन्स | बिल लेनदेन | बिल नॉन्स | बिल का भत्ता | ऐलिस से बिल की कुल आय |
|---|---|---|---|---|---|
| approve(Bill, 5) | 10 | 5 | 0 | ||
| increaseAllowance(Bill, 5) | 11 | 5+5 = 10 | 0 | ||
| transferFrom(Alice, Bill, 10) | 10,124 | 0 | 10 |
1 /**2 * @dev कॉलर द्वारा `spender` को दिए गए भत्ते को स्वचालित रूप से बढ़ाता है।3 *4 * यह {approve} का एक विकल्प है जिसका उपयोग {IERC20-approve} में वर्णित समस्याओं के शमन के रूप में किया जा सकता है।5 *6 * अद्यतनित भत्ते को इंगित करने वाला एक {अनुमोदन} इवेंट उत्सर्जित करता है।7 *8 * आवश्यकताएं:9 *10 * - `खर्च करने वाला` शून्य पता नहीं हो सकता।11 */12 function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {13 _approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));14 return true;15 }सभी दिखाएँa.add(b) फ़ंक्शन एक सुरक्षित जोड़ है। इस असंभावित मामले में कि a+b>=2^256 यह उस तरह से
लपेटता नहीं है जिस तरह से सामान्य जोड़ होता है।
12 /**3 * @dev कॉलर द्वारा `spender` को दिए गए भत्ते को स्वचालित रूप से घटाता है।4 *5 * यह {approve} का एक विकल्प है जिसका उपयोग {IERC20-approve} में वर्णित समस्याओं के शमन के रूप में किया जा सकता है।6 *7 * अद्यतनित भत्ते को इंगित करने वाला एक {अनुमोदन} इवेंट उत्सर्जित करता है।8 *9 * आवश्यकताएं:10 *11 * - `खर्च करने वाला` शून्य पता नहीं हो सकता।12 * - `खर्च करने वाले` के पास कम से कम13 * `घटाए गए मान` के कॉलर के लिए भत्ता होना चाहिए।14 */15 function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {16 _approve(_msgSender(), spender, _allowances[_msgSender()][spender].sub(subtractedValue,17 "ERC20: शून्य से नीचे घटा हुआ भत्ता"));18 return true;19 }सभी दिखाएँटोकन जानकारी को संशोधित करने वाले फ़ंक्शंस
ये चार फ़ंक्शन हैं जो वास्तविक कार्य करते हैं: _transfer, _mint, _burn, और _approve।
_transfer फ़ंक्शन
1 /**2 * @dev टोकन `राशि` को `प्रेषक` से `प्राप्तकर्ता` तक ले जाता है।3 *4 * यह आंतरिक फ़ंक्शन {ट्रांसफर} के बराबर है, और इसका उपयोग5 * उदा., स्वचालित टोकन शुल्क, स्लेशिंग तंत्र, आदि को लागू करने के लिए किया जा सकता है।6 *7 * एक {ट्रांसफर} इवेंट उत्सर्जित करता है।8 *9 * आवश्यकताएं:10 *11 * - `प्रेषक` शून्य पता नहीं हो सकता।12 * - `प्राप्तकर्ता` शून्य पता नहीं हो सकता।13 * - `प्रेषक` के पास कम से कम `राशि` का बैलेंस होना चाहिए।14 */15 function _transfer(address sender, address recipient, uint256 amount) internal virtual {सभी दिखाएँयह फ़ंक्शन, _transfer, एक खाते से दूसरे खाते में टोकन स्थानांतरित करता है। इसे दोनों
transfer (प्रेषक के अपने खाते से स्थानांतरण के लिए) और transferFrom (किसी और के खाते से स्थानांतरण के लिए भत्ते का उपयोग करने के लिए) द्वारा कॉल किया जाता है।
1 require(sender != address(0), "ERC20: transfer from the zero address");2 require(recipient != address(0), "ERC20: transfer to the zero address");एथेरियम में कोई भी वास्तव में शून्य पते का मालिक नहीं है (यानी, कोई भी ऐसी निजी कुंजी नहीं जानता है जिसकी मिलान सार्वजनिक कुंजी शून्य पते में बदल जाती है)। जब लोग उस पते का उपयोग करते हैं, तो यह आमतौर पर एक सॉफ्टवेयर बग होता है - इसलिए हम विफल हो जाते हैं यदि शून्य पते का उपयोग प्रेषक या प्राप्तकर्ता के रूप में किया जाता है।
1 _beforeTokenTransfer(sender, recipient, amount);2इस अनुबंध का उपयोग करने के दो तरीके हैं:
- इसे अपने स्वयं के कोड के लिए एक टेम्पलेट के रूप में उपयोग करें
- इससे इनहेरिट करें (opens in a new tab), और केवल उन फ़ंक्शंस को ओवरराइड करें जिन्हें आपको संशोधित करने की आवश्यकता है
दूसरा तरीका बहुत बेहतर है क्योंकि ओपनज़ेपेलिन ERC-20 कोड का पहले ही ऑडिट किया जा चुका है और इसे सुरक्षित दिखाया गया है। जब आप इनहेरिटेंस का उपयोग करते हैं तो यह स्पष्ट होता है कि आप किन फ़ंक्शंस को संशोधित करते हैं, और आपके अनुबंध पर भरोसा करने के लिए लोगों को केवल उन विशिष्ट फ़ंक्शंस का ऑडिट करने की आवश्यकता होती है।
हर बार जब टोकन का आदान-प्रदान होता है तो एक फ़ंक्शन करना अक्सर उपयोगी होता है। हालांकि, _transfer एक बहुत ही महत्वपूर्ण फ़ंक्शन है और इसे
असुरक्षित रूप से लिखना संभव है (नीचे देखें), इसलिए इसे ओवरराइड न करना सबसे अच्छा है। समाधान _beforeTokenTransfer है, एक
हुक फ़ंक्शन (opens in a new tab)। आप इस फ़ंक्शन को ओवरराइड कर सकते हैं, और इसे प्रत्येक ट्रांसफर पर कॉल किया जाएगा।
1 _balances[sender] = _balances[sender].sub(amount, "ERC20: transfer amount exceeds balance");2 _balances[recipient] = _balances[recipient].add(amount);ये वे पंक्तियाँ हैं जो वास्तव में स्थानांतरण करती हैं। ध्यान दें कि उनके बीच कुछ भी नहीं है, और हम प्राप्तकर्ता को जोड़ने से पहले प्रेषक से स्थानांतरित राशि घटाते हैं। यह महत्वपूर्ण है क्योंकि अगर बीच में किसी अलग अनुबंध को कॉल किया गया होता, तो इसका उपयोग इस अनुबंध को धोखा देने के लिए किया जा सकता था। इस तरह ट्रांसफर परमाणु होता है, इसके बीच में कुछ भी नहीं हो सकता है।
1 emit Transfer(sender, recipient, amount);2 }अंत में, एक ट्रांसफर इवेंट उत्सर्जित करें। इवेंट स्मार्ट अनुबंधों के लिए सुलभ नहीं हैं, लेकिन ब्लॉकचेन के बाहर चलने वाला कोड
इवेंट को सुन सकता है और उन पर प्रतिक्रिया कर सकता है। उदाहरण के लिए, एक वॉलेट ट्रैक रख सकता है कि मालिक को अधिक टोकन कब मिलते हैं।
_mint और _burn फ़ंक्शंस
ये दो फ़ंक्शन (_mint और _burn) टोकन की कुल आपूर्ति को संशोधित करते हैं।
वे आंतरिक हैं और इस अनुबंध में उन्हें कॉल करने वाला कोई फ़ंक्शन नहीं है,
इसलिए वे केवल तभी उपयोगी होते हैं जब आप अनुबंध से इनहेरिट करते हैं और यह तय करने के लिए अपना स्वयं का
तर्क जोड़ते हैं कि किन परिस्थितियों में नए टोकन मिंट करने हैं या मौजूदा
को बर्न करना है।
ध्यान दें: प्रत्येक ERC-20 टोकन का अपना व्यावसायिक तर्क होता है जो टोकन प्रबंधन को निर्धारित करता है।
उदाहरण के लिए, एक निश्चित आपूर्ति अनुबंध केवल कंस्ट्रक्टर में
_mint को कॉल कर सकता है और कभी भी _burn को कॉल नहीं कर सकता है। एक अनुबंध जो टोकन बेचता है,
जब इसका भुगतान किया जाता है तो _mint को कॉल करेगा, और संभवतः किसी बिंदु पर _burn को कॉल करेगा
ताकि भगोड़ा मुद्रास्फीति से बचा जा सके।
1 /** @dev `राशि` टोकन बनाता है और उन्हें `खाते` को सौंपता है, जिससे2 * कुल आपूर्ति बढ़ जाती है।3 *4 * `from` को शून्य पते पर सेट करके एक {Transfer} इवेंट उत्सर्जित करता है।5 *6 * आवश्यकताएं:7 *8 * - `to` शून्य पता नहीं हो सकता है।9 */10 function _mint(address account, uint256 amount) internal virtual {11 require(account != address(0), "ERC20: mint to the zero address");12 _beforeTokenTransfer(address(0), account, amount);13 _totalSupply = _totalSupply.add(amount);14 _balances[account] = _balances[account].add(amount);15 emit Transfer(address(0), account, amount);16 }सभी दिखाएँजब टोकन की कुल संख्या बदलती है तो _totalSupply को अपडेट करना सुनिश्चित करें।
1 /**2 * @dev `खाते` से `राशि` टोकन को नष्ट कर देता है, जिससे3 * कुल आपूर्ति कम हो जाती है।4 *5 * `to` को शून्य पते पर सेट करके एक {Transfer} इवेंट उत्सर्जित करता है।6 *7 * आवश्यकताएं:8 *9 * - `खाता` शून्य पता नहीं हो सकता है।10 * - `खाते` में कम से कम `राशि` टोकन होने चाहिए।11 */12 function _burn(address account, uint256 amount) internal virtual {13 require(account != address(0), "ERC20: burn from the zero address");1415 _beforeTokenTransfer(account, address(0), amount);1617 _balances[account] = _balances[account].sub(amount, "ERC20: burn amount exceeds balance");18 _totalSupply = _totalSupply.sub(amount);19 emit Transfer(account, address(0), amount);20 }सभी दिखाएँ_burn फ़ंक्शन लगभग _mint के समान है, सिवाय इसके कि यह दूसरी दिशा में जाता है।
_approve फ़ंक्शन
यह वह फ़ंक्शन है जो वास्तव में अलाउंस निर्दिष्ट करता है। ध्यान दें कि यह एक मालिक को एक भत्ता निर्दिष्ट करने की अनुमति देता है जो मालिक के वर्तमान बैलेंस से अधिक है। यह ठीक है क्योंकि बैलेंस ट्रांसफर के समय जांचा जाता है, जब यह अलाउंस बनाए जाने पर बैलेंस से अलग हो सकता है।
1 /**2 * @dev `मालिक` के टोकन पर `खर्च करने वाले` के भत्ते के रूप में `राशि` सेट करता है।3 *4 * यह आंतरिक फ़ंक्शन `approve` के बराबर है, और इसका उपयोग5 * उदा., कुछ सबसिस्टम के लिए स्वचालित भत्ते सेट करने आदि के लिए किया जा सकता है।6 *7 * एक {Approval} इवेंट उत्सर्जित करता है।8 *9 * आवश्यकताएं:10 *11 * - `मालिक` शून्य पता नहीं हो सकता।12 * - `खर्च करने वाला` शून्य पता नहीं हो सकता।13 */14 function _approve(address owner, address spender, uint256 amount) internal virtual {15 require(owner != address(0), "ERC20: approve from the zero address");16 require(spender != address(0), "ERC20: approve to the zero address");1718 _allowances[owner][spender] = amount;सभी दिखाएँ
अनुमोदन इवेंट उत्सर्जित करें। एप्लिकेशन कैसे लिखा जाता है, इसके आधार पर, खर्च करने वाले अनुबंध को
अनुमोदन के बारे में या तो मालिक द्वारा या इन घटनाओं को सुनने वाले सर्वर द्वारा बताया जा सकता है।
1 emit Approval(owner, spender, amount);2 }3दशमलव वैरिएबल को संशोधित करें
123 /**4 * @dev {दशमलव} को 18 के डिफ़ॉल्ट मान के अलावा किसी अन्य मान पर सेट करता है।5 *6 * चेतावनी: इस फ़ंक्शन को केवल कंस्ट्रक्टर से ही कॉल किया जाना चाहिए। अधिकांश7 * एप्लिकेशन जो टोकन अनुबंधों के साथ इंटरैक्ट करते हैं, वे8 * {दशमलव} के कभी भी बदलने की उम्मीद नहीं करेंगे, और यदि ऐसा होता है तो वे गलत तरीके से काम कर सकते हैं।9 */10 function _setupDecimals(uint8 decimals_) internal {11 _decimals = decimals_;12 }सभी दिखाएँयह फ़ंक्शन _decimals वैरिएबल को संशोधित करता है जिसका उपयोग उपयोगकर्ता इंटरफ़ेस को यह बताने के लिए किया जाता है कि राशि की व्याख्या कैसे करें।
आपको इसे कंस्ट्रक्टर से कॉल करना चाहिए। किसी भी बाद के बिंदु पर इसे कॉल करना बेईमानी होगी, और एप्लिकेशन
इसे संभालने के लिए डिज़ाइन नहीं किए गए हैं।
हुक्स
12 /**3 * @dev टोकन के किसी भी हस्तांतरण से पहले कॉल किया जाने वाला हुक। इसमें4 * मिंटिंग और बर्निंग शामिल है।5 *6 * कॉलिंग की शर्तें:7 *8 * - जब `from` और `to` दोनों गैर-शून्य हों, तो ``from`` के टोकन की `राशि`9 * `to` को हस्तांतरित की जाएगी।10 * - जब `from` शून्य हो, तो `to` के लिए `राशि` टोकन बनाए जाएंगे।11 * - जब `to` शून्य हो, तो ``from`` के टोकन की `राशि` जला दी जाएगी।12 * - `from` और `to` दोनों कभी भी शून्य नहीं होते हैं।13 *14 * हुक के बारे में अधिक जानने के लिए, xref:ROOT:extending-contracts.adoc#using-hooks[हुक का उपयोग करना] पर जाएं।15 */16 function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual { }17}सभी दिखाएँयह ट्रांसफर के दौरान कॉल किया जाने वाला हुक फ़ंक्शन है। यह यहाँ खाली है, लेकिन अगर आपको इसकी आवश्यकता है तो आप इसे बस ओवरराइड करें।
निष्कर्ष
समीक्षा के लिए, यहां इस अनुबंध में कुछ सबसे महत्वपूर्ण विचार दिए गए हैं (मेरी राय में, आपका अलग होने की संभावना है):
- यह कई मामलों में महत्वपूर्ण है, लेकिन यहां नहीं।ब्लॉकचेन पर कोई रहस्य नहीं हैं। कोई भी जानकारी जिसे एक स्मार्ट अनुबंध एक्सेस कर सकता है, पूरी दुनिया के लिए उपलब्ध है।
- आप अपने स्वयं के लेनदेन के क्रम को नियंत्रित कर सकते हैं, लेकिन यह नहीं कि दूसरे लोगों का लेनदेन कब होता है। यही कारण है कि एक भत्ता बदलना खतरनाक हो सकता है, क्योंकि यह खर्च करने वाले को दोनों भत्तों का योग खर्च करने देता है।
uint256प्रकार के मान रैप अराउंड होते हैं। दूसरे शब्दों में, 0-1=2^256-1। यदि वह वांछित व्यवहार नहीं है, तो आपको इसकी जांच करनी होगी (या SafeMath लाइब्रेरी का उपयोग करें जो आपके लिए ऐसा करती है)। ध्यान दें कि यह सॉलिडिटी 0.8.0 (opens in a new tab) में बदल गया।- एक विशिष्ट प्रकार के सभी स्टेट परिवर्तन एक विशिष्ट स्थान पर करें, क्योंकि इससे ऑडिटिंग आसान हो जाती है।
यही कारण है कि हमारे पास, उदाहरण के लिए,
_approveहै, जिसेapprove,transferFrom,increaseAllowance, औरdecreaseAllowanceद्वारा कॉल किया जाता है। - स्टेट परिवर्तन परमाणु होने चाहिए, उनके बीच कोई अन्य क्रिया नहीं होनी चाहिए (जैसा कि आप
_transferमें देख सकते हैं)। ऐसा इसलिए है क्योंकि स्टेट परिवर्तन के दौरान आपकी एक असंगत स्टेट होती है। उदाहरण के लिए, प्रेषक के बैलेंस से कटौती करने और प्राप्तकर्ता के बैलेंस में जोड़ने के समय के बीच, अस्तित्व में होने चाहिए उससे कम टोकन होते हैं। इसका संभावित रूप से दुरुपयोग किया जा सकता है यदि उनके बीच संचालन हों, विशेष रूप से एक अलग अनुबंध को कॉल।
अब जब आपने देख लिया है कि ओपनज़ेपेलिन ERC-20 अनुबंध कैसे लिखा जाता है, और विशेष रूप से इसे कैसे अधिक सुरक्षित बनाया जाता है, तो जाएं और अपने स्वयं के सुरक्षित अनुबंध और एप्लिकेशन लिखें।
मेरे और काम के लिए यहाँ देखें (opens in a new tab)।
पेज का अंतिम अपडेट: 3 मार्च 2026
