昨天我們講了 ERC20 的其中一半的 function 怎麼實作,今天就把剩下的補齊吧!
另外,我們發現 transfer
跟 transferFrom
有幾乎相同的程式碼結構,因此把這兩個 function 中的一段程式碼抽出來成為一個新的 function _transfer
,這樣的技巧其實在開發中非常常見,如果他用得更平凡可以變成一個 library 來抽離呢!
本日合約:
pragma solidity ^0.4.25;
import "browser/SafeMath.sol";
import "browser/IERC20.sol";
contract ERC20 is IERC20 {
using SafeMath for uint256;
uint256 private _totalSupply;
mapping(address => uint256) _balances;
mapping(address => mapping(address => uint256)) _approve;
// 所有存在的 Token 數量
function totalSupply() external view returns (uint256) {
return _totalSupply;
}
// 讀取 tokenOwner 這個 address 所持有的 Token 數量
// address => uint256
function balanceOf(address tokenOwner) external view returns (uint256 balance) {
return _balances[tokenOwner];
}
// 從 msg.sender 轉 tokens 個 Token 給 to 這個 address
// msg.sender ---tokens---> to
function transfer(address to, uint256 tokens) external returns (bool success) {
return _transfer(msg.sender, to, tokens);
}
// 得到 tokenOwner 授權給 spender 使用的 Token 剩餘數量
function allowance(address tokenOwner, address spender) external view returns (uint256 remaining) {
return _approve[tokenOwner][spender];
}
// tokenOwner -> spender -> tokens
// address => address => uint256
// msg.sender 授權給 spender 可使用自己的 tokens 個 Token
function approve(address spender, uint256 tokens) external returns (bool success) {
_approve[msg.sender][spender] = tokens;
emit Approval(msg.sender, spender, tokens);
return true;
}
// 將 tokens 個 Token 從 from 轉到 to
function transferFrom(address from, address to, uint256 tokens) external returns (bool success) {
_approve[from][msg.sender] = _approve[from][msg.sender].sub(tokens);
return _transfer(from, to, tokens);
}
function _transfer(address from, address to, uint256 tokens) internal returns (bool success) {
_balances[from] = _balances[from].sub(tokens);
_balances[to] = _balances[to].add(tokens);
emit Transfer(from, to, tokens);
return true;
}
}
本日影片:
https://youtu.be/-FmR_oIOwtE
Smart Contract 實戰教學播放清單:
https://www.youtube.com/playlist?list=PLHmOMPRfmOxSJcrlwyandWYiuP9ZAMYoF