ERC-20 토큰 스마트 계약 이해하기
이더리움에서 가장 대표적인 스마트 계약 표준 중 하나인 ERC-20은 이더리움 블록체인 상에서 대체 가능 토큰 구현을 위한 모든 스마트 계약의 기술적 표준으로 등장했습니다.
ERC-20은 모든 대체 가능한 이더리움 토큰이 준수해야 하는 공통 규칙 목록을 정의합니다. 따라서 이 토큰 표준을 통해 모든 유형의 개발자는 새로운 토큰이 더 큰 이더리움 시스템 내에서 어떻게 작동할지 정확하게 예측할 수 있습니다. 토큰이 규정을 따르는 한, 개발자들은 새로운 토큰이 출시될 때마다 프로젝트를 다시 작업할 필요가 없으므로 작업이 더 간단하고 쉬워집니다.
다음은 ERC-20이 구현해야 하는 함수를 인터페이스로 나타낸 것입니다. 인터페이스가 무엇인지 확실하지 않다면 솔리디티의 객체 지향 프로그래밍(OOP) (opens in a new tab)에 대한 저희 아티클을 확인해 보세요.
1pragma solidity ^0.6.0;23interface IERC20 {45 function totalSupply() external view returns (uint256);6 function balanceOf(address account) external view returns (uint256);7 function allowance(address owner, address spender) external view returns (uint256);89 function transfer(address recipient, uint256 amount) external returns (bool);10 function approve(address spender, uint256 amount) external returns (bool);11 function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);121314 event Transfer(address indexed from, address indexed to, uint256 value);15 event Approval(address indexed owner, address indexed spender, uint256 value);16}모두 보기여기에서는 각 줄 별로 각각의 함수가 어떤 역할을 하는지 설명합니다. 이어서 ERC-20 토큰의 간단한 구현을 제시합니다.
게터
1function totalSupply() external view returns (uint256);존재하는 토큰의 양을 반환합니다. 이 함수는 게터(getter)이며 컨트랙트의 상태를 수정하지 않습니다. 솔리디티에는 부동 소수점이 없다는 점에 유의하세요. 따라서 대부분의 토큰은 18개의 소수 자릿수를 사용하며, 1 토큰 = 1000000000000000000으로 취급하여 총공급량 및 기타 결과를 반환합니다. 모든 토큰이 18개의 소수 자릿수를 사용하는 것은 아니므로, 토큰을 다룰 때 이 점에 유의해야 합니다.
1function balanceOf(address account) external view returns (uint256);특정 주소(account)가 소유한 토큰의 양을 반환합니다. 이 함수는 게터(getter)이며 컨트랙트의 상태를 수정하지 않습니다.
1function allowance(address owner, address spender) external view returns (uint256);ERC-20 표준에서는 한 주소가 다른 주소에게 자신의 주소에서 토큰을 인출할 수 있는 허용량(allowance)을 부여할 수 있습니다. 이 게터는 spender가 owner를 대신하여 사용할 수 있도록 허용된 토큰의 남은 수량을 반환합니다. 이 함수는 게터이며 컨트랙트의 상태를 수정하지 않고, 기본적으로 0을 반환해야 합니다.
함수
1function transfer(address recipient, uint256 amount) external returns (bool);amount 만큼의 토큰을 함수 호출자 주소(msg.sender)에서 수신자 주소로 전송합니다. 이 함수는 나중에 정의될 Transfer 이벤트를 발생시킵니다. 전송이 가능했다면 true를 반환합니다.
1function approve(address spender, uint256 amount) external returns (bool);spender가 함수 호출자(msg.sender)의 잔액에서 전송하도록 허용된 allowance(허용량)를 설정합니다. 이 함수는 Approval 이벤트를 발생시킵니다. 이 함수는 허용량이 성공적으로 설정되었는지 여부를 반환합니다.
1function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);허용량 메커니즘을 사용하여 sender에서 recipient로 amount 만큼의 토큰을 이동시킵니다. amount는 호출자의 허용량에서 차감됩니다. 이 함수는 Transfer 이벤트를 발생시킵니다.
이벤트
1event Transfer(address indexed from, address indexed to, uint256 value);이 이벤트는 토큰의 양(value)이 from 주소에서 to 주소로 전송될 때 발생합니다.
새로운 토큰이 발행되는 경우 일반적으로 보내는 주소 from이 0x00..0000 이 되며, 소각되는 경우 받는 주소 to가 0x00..0000 이 됩니다.
1event Approval(address indexed owner, address indexed spender, uint256 value);이 이벤트는 spender가 사용할 토큰의 양(value)이 owner에 의해 승인될 때 발생합니다.
ERC-20 토큰의 기본 구현
아래는 ERC-20 토큰을 구성하는 가장 기본적인 코드입니다.
1pragma solidity ^0.8.0;23interface IERC20 {45 function totalSupply() external view returns (uint256);6 function balanceOf(address account) external view returns (uint256);7 function allowance(address owner, address spender) external view returns (uint256);89 function transfer(address recipient, uint256 amount) external returns (bool);10 function approve(address spender, uint256 amount) external returns (bool);11 function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);121314 event Transfer(address indexed from, address indexed to, uint256 value);15 event Approval(address indexed owner, address indexed spender, uint256 value);16}171819contract ERC20Basic is IERC20 {2021 string public constant name = "ERC20Basic";22 string public constant symbol = "ERC";23 uint8 public constant decimals = 18;242526 mapping(address => uint256) balances;2728 mapping(address => mapping (address => uint256)) allowed;2930 uint256 totalSupply_ = 10 ether;313233 constructor() {34 balances[msg.sender] = totalSupply_;35 }3637 function totalSupply() public override view returns (uint256) {38 return totalSupply_;39 }4041 function balanceOf(address tokenOwner) public override view returns (uint256) {42 return balances[tokenOwner];43 }4445 function transfer(address receiver, uint256 numTokens) public override returns (bool) {46 require(numTokens <= balances[msg.sender]);47 balances[msg.sender] = balances[msg.sender]-numTokens;48 balances[receiver] = balances[receiver]+numTokens;49 emit Transfer(msg.sender, receiver, numTokens);50 return true;51 }5253 function approve(address delegate, uint256 numTokens) public override returns (bool) {54 allowed[msg.sender][delegate] = numTokens;55 emit Approval(msg.sender, delegate, numTokens);56 return true;57 }5859 function allowance(address owner, address delegate) public override view returns (uint) {60 return allowed[owner][delegate];61 }6263 function transferFrom(address owner, address buyer, uint256 numTokens) public override returns (bool) {64 require(numTokens <= balances[owner]);65 require(numTokens <= allowed[owner][msg.sender]);6667 balances[owner] = balances[owner]-numTokens;68 allowed[owner][msg.sender] = allowed[owner][msg.sender]-numTokens;69 balances[buyer] = balances[buyer]+numTokens;70 emit Transfer(owner, buyer, numTokens);71 return true;72 }73}모두 보기ERC-20 토큰 표준의 또 다른 훌륭한 구현은 OpenZeppelin ERC-20 구현 (opens in a new tab)입니다.
페이지 마지막 업데이트됨: 2025년 8월 21일