美文网首页
TokenERC20代币详情

TokenERC20代币详情

作者: 吴敬悦 | 来源:发表于2018-11-23 16:08 被阅读146次

pragma solidity ^0.4.20;

interface tokenRecipient { function receiveApproval(address _from, uint256 _value, address _token, bytes _extraData) external; }

contract TokenERC20 {

    string public name;          // 代币名称

    string public symbol;        // 代币符号

    uint8 public decimals = 18;  // decimals 可以有的小数点个数,最小的代币单位。18 是建议的默认值

    uint256 public totalSupply;  // 代币总金额

    // 用mapping保存每个地址对应的余额

    mapping (address => uint256) public balanceOf;

    // 存储对账号的控制(授权人地址,自己地址,授权金额)

    mapping (address => mapping (address => uint256)) public allowance;

    // 事件,用来通知客户端交易发生

    event Transfer(address indexed from, address indexed to, uint256 value);

    // 事件,用来通知客户端代币被消费

    event Burn(address indexed from, uint256 value);

    /**

    * 初始化构造

    */

    constructor(uint256 initialSupply, string tokenName, string tokenSymbol) public {

        totalSupply = initialSupply * 10 ** uint256(decimals);  // 供应的份额,份额跟最小的代币单位有关,份额 = 币数 * 10 ** decimals。

        balanceOf[msg.sender] = totalSupply;                // 创建者拥有所有的代币

        name = tokenName;                                  // 代币名称

        symbol = tokenSymbol;                              // 代币符号

    }

    /**

    * 代币交易转移的内部实现

    */

    function _transfer(address _from, address _to, uint _value) internal {

        // 确保目标地址不为0x0,因为0x0地址代表销毁

        require(_to != 0x0);

        // 检查发送者余额

        require(balanceOf[_from] >= _value);

        // 确保转移为正数个

        require(balanceOf[_to] + _value > balanceOf[_to]);

        // 以下用来检查交易,

        uint previousBalances = balanceOf[_from] + balanceOf[_to];

        // 转账方(付款方)自己账本金额减少_value

        balanceOf[_from] -= _value;

        // 收款方账本金额增加_value

        balanceOf[_to] += _value;

        emit Transfer(_from, _to, _value);

        // 用assert来检查代码逻辑。

        assert(balanceOf[_from] + balanceOf[_to] == previousBalances);

    }

    /**

    * 代币交易转移

    * 从你的账户发送`_value`个代币到 `_to`账号

    *

    * @param _to 接收者地址

    * @param _value 转移数额

    */

    function transfer(address _to, uint256 _value) public {//这个是使用自己钱包的余额

        _transfer(msg.sender, _to, _value);

    }

    /**

    * 账号之间代币交易转移

    * @param _from 调用者授权方地址

    * @param _to 接收者地址

    * @param _value 转移数额

    */

    function transferFrom(address _from, address _to, uint256 _value) public returns (bool success) {//这个是授权余额

        require(_value <= allowance[_from][msg.sender]);    // 验证授权金额是否大于所交易金额

        allowance[_from][msg.sender] -= _value;

        _transfer(_from, _to, _value);

        return true;

    }

    /**

    * 设置某个地址(合约)可以以交易者的名义花费的代币数。

    *

    * 允许发送者`_spender` 花费不多于 `_value` 个代币

    *

    * @param _spender The address authorized to spend

    * @param _value the max amount they can spend

    */

    function approve(address _spender, uint256 _value) public

        returns (bool success) {

        allowance[msg.sender][_spender] = _value;

        return true;

    }

    /**

    * 设置允许一个地址(合约)以交易者名义可最多花费的代币数。

    *

    * @param _spender 被授权的地址(合约)

    * @param _value 最大可花费代币数

    * @param _extraData 发送给合约的附加数据

    */

    function approveAndCall(address _spender, uint256 _value, bytes _extraData)

        public

        returns (bool success) {

        tokenRecipient spender = tokenRecipient(_spender);

        if (approve(_spender, _value)) {

            spender.receiveApproval(msg.sender, _value, this, _extraData);

            return true;

        }

    }

    /**

    * 销毁创建者账户中指定个代币

    */

    function burn(uint256 _value) public returns (bool success) {

        require(balanceOf[msg.sender] >= _value);  // 检查自己账本余额是否充足(大于_value)

        balanceOf[msg.sender] -= _value;            // 满足的话就减去对应值(_value)

        totalSupply -= _value;                      // 总金额也减少

        emit Burn(msg.sender, _value);

        return true;

    }

    /**

    * 销毁用户账户中指定个代币

    *

    * Remove `_value` tokens from the system irreversibly on behalf of `_from`.

    *

    * @param _from the address of the sender

    * @param _value the amount of money to burn

    */

    function burnFrom(address _from, uint256 _value) public returns (bool success) {

        require(balanceOf[_from] >= _value);                // Check if the targeted balance is enough

        require(_value <= allowance[_from][msg.sender]);    // Check allowance

        balanceOf[_from] -= _value;                        // Subtract from the targeted balance

        allowance[_from][msg.sender] -= _value;            // Subtract from the sender's allowance

        totalSupply -= _value;                              // Update totalSupply

        emit Burn(_from, _value);

        return true;

    }

}

每一个账户有两份代币额度,不同的代币需要使用不同的方法进行转账:

账户的两个转账函数

相关文章

网友评论

      本文标题:TokenERC20代币详情

      本文链接:https://www.haomeiwen.com/subject/gaijqqtx.html