实现一个可管理、增发、兑换、冻结等高级功能的代币

转自 : https://learnblockchain.cn/2018/01/27/create-token2/

实现代币的管理者

虽然区块链是去中心化的,但是实现对代币(合约)的管理,也在许多应用中有需求,为了对代币进行管理,首先需要给合约添加一个管理者。

我们来看看如果实现,先创建一个owned合约。

contract owned {
 address public owner;

 function owned() {
 owner = msg.sender;
 }

 modifier onlyOwner {
 require(msg.sender == owner);
 _;
 }

 // 实现所有权转移
 function transferOwnership(address newOwner) onlyOwner {
 owner = newOwner;
 }
}

这个合约重要的是加入了一个函数修改器(Function Modifiers)onlyOwner,函数修改器是一个合约属性,可以被继承,还能被重写。它用于在函数执行前检查某种前置条件。
关于函数修改器可进一步阅读Solidity 教程系列10 - 完全理解函数修改器

如果熟悉Python的同学,会发现函数修改器的作用和Python的装饰器很相似。

然后让代币合约继承owned以拥有onlyOwner修改器,代码如下:

contract MyToken is owned {
 function MyToken(
 uint256 initialSupply,
 string tokenName,
 uint8 decimalUnits,
 string tokenSymbol,
 address centralMinter
 ) {
 if(centralMinter != 0 ) owner = centralMinter;
 }
}

代币增发

实现代币增发,代币增发就如同央行印钞票一样,想必很多人都需要这样的功能。

给合约添加以下的方法:

function mintToken(address target, uint256 mintedAmount) onlyOwner {
 balanceOf[target] += mintedAmount;
 totalSupply += mintedAmount;
 Transfer(0, owner, mintedAmount);
 Transfer(owner, target, mintedAmount);
 }

注意onlyOwner修改器添加在函数末尾,这表示只有ower才能调用这用函数。
他的功能很简单,就是给指定的账户增加代币,同时增加总供应量。

资产冻结

有时为了监管的需要,需要实现冻结某些账户,冻结后,其资产仍在账户,但是不允许交易,之道解除冻结。
给合约添加以下的变量和方法(可以添加到合约的任何地方,但是建议把mapping加到和其他mapping一起,event也是如此):

mapping (address => bool) public frozenAccount;
event FrozenFunds(address target, bool frozen);

function freezeAccount(address target, bool freeze) onlyOwner {
 frozenAccount[target] = freeze;
 FrozenFunds(target, freeze);
}

单单以上的代码还无法冻结,需要把他加入到transfer函数中才能真正生效,因此修改transfer函数

function transfer(address _to, uint256 _value) {
 require(!frozenAccount[msg.sender]);
 ...
}

这样在转账前,对发起交易的账号做一次检查,只有不是被冻结的账号才能转账。

代币买卖(兑换)

可以自己的货币中实现代币与其他数字货币(ether 或其他tokens)的兑换机制。有了这个功能,我们的合约就可以在一买一卖中赚利润了。

先来设置下买卖价格

uint256 public sellPrice;
uint256 public buyPrice;

function setPrices(uint256 newSellPrice, uint256 newBuyPrice) onlyOwner {
 sellPrice = newSellPrice;
 buyPrice = newBuyPrice;
}

setPrices()添加了onlyOwner修改器,注意买卖的价格单位是wei(最小的货币单位: 1 eth = 1000000000000000000 wei)

添加来添加买卖函数:

function buy() payable returns (uint amount){
 amount = msg.value / buyPrice;                    // calculates the amount
 require(balanceOf[this] >= amount);               // checks if it has enough to sell
 balanceOf[msg.sender] += amount;                  // adds the amount to buyer's balance
 balanceOf[this] -= amount;                        // subtracts amount from seller's balance
 Transfer(this, msg.sender, amount);               // execute an event reflecting the change
 return amount;                                    // ends function and returns
}

function sell(uint amount) returns (uint revenue){
 require(balanceOf[msg.sender] >= amount);         // checks if the sender has enough to sell
 balanceOf[this] += amount;                        // adds the amount to owner's balance
 balanceOf[msg.sender] -= amount;                  // subtracts the amount from seller's balance
 revenue = amount * sellPrice;
 msg.sender.transfer(revenue);                     // sends ether to the seller: it's important to do this last to prevent recursion attacks
 Transfer(msg.sender, this, amount);               // executes an event reflecting on the change
 return revenue;                                   // ends function and returns
}

加入了买卖功能后,要求我们在创建合约时发送足够的以太币,以便合约有能力回购市面上的代币,否则合约将破产,用户没法先合约卖代币。

实现Gas的自动补充

以太坊中的交易时需要gas(支付给矿工的费用,费用以ether来支付)。而如果用户没有以太币,只有代币的情况(或者我们想向用户隐藏以太坊的细节),就需要自动补充gas的功能。这个功能将使我们代币更加好用。

自动补充的逻辑是这样了,在执行交易之前,我们判断用户的余额(用来支付矿工的费用),如果用户的余额非常少(低于某个阈值时)可能影响到交易进行,合约自动售出一部分代币来补充余额,以帮助用户顺利完成交易。

先来设定余额阈值:

uint minBalanceForAccounts;

 function setMinBalance(uint minimumBalanceInFinney) onlyOwner {
 minBalanceForAccounts = minimumBalanceInFinney * 1 finney;
 }

finney 是货币单位 1 finney = 0.001eth
然后交易中加入对用户的余额的判断。

function transfer(address _to, uint256 _value) {
 ...
 if(msg.sender.balance < minBalanceForAccounts)
 sell((minBalanceForAccounts - msg.sender.balance) / sellPrice);
 if(_to.balance<minBalanceForAccounts)   // 可选,让接受者也补充余额,以便接受者使用代币。
 _to.send(sell((minBalanceForAccounts - _to.balance) / sellPrice));
}

代码部署

高级功能完整代码请前往我的小专栏, 项目的完整的部署方法参考上一篇,不同的是创建合约时需要预存余额,如图:

专栏已经有多篇文章介绍Remix Solidity IDE的使用,这里就不一一截图演示了,请大家自己测试验证。

如果你在创建代币的过程中遇到问题,欢迎到我的知识星球提问,作为星球成员福利,成员可加入区块链技术付费交流群。

参考文档

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容