通用模式

翻译原文

date:20170803

从合约中取回钱币

发送钱币的推荐方法是使用取回模式。尽管发送钱币最简单的简单方法是直接调用send。但是它会引入潜在的风险,所以也不推荐使用。你可以阅读安全考虑章节,了解更多信息。
这是一个在合约中使用取回模式的例子,功能是发送最多的钱币到合约中,来成为首富,灵感来自King of the ether
在下面的合约中,如果你要篡夺成为首富,你要接收去世的人的财产来成为新的首富。

pragma solidity ^0.4.11;

contract WithdrawalContract {
    address public richest;
    uint public mostSent;

    mapping (address => uint) pendingWithdrawals;

    function WithdrawalContract() payable {
        richest = msg.sender;
        mostSent = msg.value;
    }

    function becomeRichest() payable returns (bool) {
        if (msg.value > mostSent) {
            pendingWithdrawals[richest] += msg.value;
            richest = msg.sender;
            mostSent = msg.value;
            return true;
        } else {
            return false;
        }
    }

    function withdraw() {
        uint amount = pendingWithdrawals[msg.sender];
        // 记得在发送之前要把余额设置为0,以防止重入攻击
        pendingWithdrawals[msg.sender] = 0;
        msg.sender.transfer(amount);
    }
}

这和直接发送模式相反:

pragma solidity ^0.4.11;

contract SendContract {
    address public richest;
    uint public mostSent;

    function SendContract() payable {
        richest = msg.sender;
        mostSent = msg.value;
    }

    function becomeRichest() payable returns (bool) {
        if (msg.value > mostSent) {
            // 这一行会出现问题(详情在下文解释)
            richest.transfer(msg.value);
            richest = msg.sender;
            mostSent = msg.value;
            return true;
        } else {
            return false;
        }
    }
}

注意,在这个例子中,攻击者可以困住合约,让合约处于不可使用的状态。通过让richest成为一个合约的地址,该合约有一个回调函数,但是这个回调会执行失败(例如使用revert()或者只是消耗2300 以上的gas),这样,通过transfer调用分发钱币到”有毒“的合约中,它就会失败,因此,becomeRichest就会失败,让合约永远不能正常执行。
相反,如果你使用第一个例子取回模式,那么攻击者只能让他或她的取回函数失败,但是合约的其余代码都没有问题。

限制访问

限制访问时合约的通用模式。注意你不能限制任何人或者计算机读取你的交易内容或者你合约状态。你可以用加密提升难度。但是如果你的合约是可以读取数据的,那么所有人都可以读取数据。
你可以通过其他合约,来限制合约的访问。默认情况下是这样的,除非你把状态变量声明为public
另外,你可以限制谁可以修改你的合约代码,或者调用你的合约函数,这是这个章节要说的:
使用函数修改器使得限制高度可读。

pragma solidity ^0.4.11;

contract AccessRestriction {
    // 这些变量在构造期间执行,
    // `msg.sender`是创建该合约的账号
    address public owner = msg.sender;
    uint public creationTime = now;

    // 修改器可以用来修改函数体。
    // 如果使用了这个修改器,它会先检测,并且只有函数从特定地址调用才会放行
    modifier onlyBy(address _account)
    {
        require(msg.sender == _account);
        // 不要忘了"_;"!
        // 它代表了使用该修改器的函数的函数体
        _;
    }

    /// 让 `_newOwner` 成为该合约的拥有者
    function changeOwner(address _newOwner)
        onlyBy(owner)
    {
        owner = _newOwner;
    }

    modifier onlyAfter(uint _time) {
        require(now >= _time);
        _;
    }

    /// 擦除拥有者信息。
    /// 只能在合约创建6个星期之后被调用。
    function disown()
        onlyBy(owner)
        onlyAfter(creationTime + 6 weeks)
    {
        delete owner;
    }

    // 这个修改器要求函数调用要有一定数量的钱币
    // 如果调用者发送了太多钱币,会被退还,但是会在函数体执行之后,再退还
    // Solidity v0.4.0之前这么做是很危险的,因为可能会忽略`_;`之后的版本。
    modifier costs(uint _amount) {
        require(msg.value >= _amount);
        _;
        if (msg.value > _amount)
            msg.sender.send(msg.value - _amount);
    }

    function forceOwnerChange(address _newOwner)
        costs(200 ether)
    {
        owner = _newOwner;
        // 一些示例条件
        if (uint(owner) & 0 == 1)
            // Solidity  v0.4.0之前不会退还.
            return;
        // 返回多余的钱币
    }
}

访问控制更加特殊的方法会在下个例子中讨论。

状态机

合约通常像一个状态机,这意味着他们有特定的阶段,不同的阶段表现不同或者会调用不同的函数。一个函数调用可以结束一个阶段,让合约进入到另一个阶段(尤其是合约模型是可交互的)。有些阶段在某个时间点会自动达到,也是很普遍的。

一个例子是秘密竞价合约,在”接收秘密竞价“的时候开始,然后转换到”披露竞价“阶段,然后在确定竞价结果的阶段结束。

函数修改器可以用在这种场合中,来改变状态,以及保护合约的不正当使用。

例子

在下面的例子中,修改器atStage保证了函数只能在特定的时候被调用。

自动计时交易通过timeTransitions修改器来处理,可以用在任意函数中。

注意:修改器顺序很重要。如果atStage和timedTransitions结合起来,你要保证atStage在最后面,然后合约会进入新的阶段。

最后,当函数结束的时候,transitionNext修改器会自动的用来进入下个阶段。

注意:修改器可能会被忽略。在Solidity v0.4.0之前可能会发生:因为修改器只是简单的替换代码,而不是使用函数调用,所以如果函数return掉的时候,transitionNext修改器中的代码可能会被忽略。 如果你要这么做,那么你应该在这些函数中手动调用nextStage。从版本0.4.0开始,修改器代码即使函数return了,也能够执行。

pragma solidity ^0.4.11;

contract StateMachine {
    enum Stages {
        AcceptingBlindedBids,
        RevealBids,
        AnotherStage,
        AreWeDoneYet,
        Finished
    }

    // 这是当前状态。
    Stages public stage = Stages.AcceptingBlindedBids;

    uint public creationTime = now;

    modifier atStage(Stages _stage) {
        require(stage == _stage);
        _;
    }

    function nextStage() internal {
        stage = Stages(uint(stage) + 1);
    }

    // 实现阶段过渡。要确保这个修改器放在最前面
    // 否则不会进入下个阶段
    modifier timedTransitions() {
        if (stage == Stages.AcceptingBlindedBids &&
                    now >= creationTime + 10 days)
            nextStage();
        if (stage == Stages.RevealBids &&
                now >= creationTime + 12 days)
            nextStage();
        // 通过交易进入下一个阶段
        _;
    }

    // 修改器的顺序在这里是非常重要的
    function bid()
        payable
        timedTransitions
        atStage(Stages.AcceptingBlindedBids)
    {
        // 这里我们不会先实现。
    }

    function reveal()
        timedTransitions
        atStage(Stages.RevealBids)
    {
    }

    // 这个修改器会使得函数结束的时候进入下一个阶段
    modifier transitionNext()
    {
        _;
        nextStage();
    }

    function g()
        timedTransitions
        atStage(Stages.AnotherStage)
        transitionNext
    {
    }

    function h()
        timedTransitions
        atStage(Stages.AreWeDoneYet)
        transitionNext
    {
    }

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

推荐阅读更多精彩内容

  • 作者:Soroush Khanlou,原文链接,原文日期:2016/8/8译者:Cwift;校对:Crystal ...
    梁杰_numbbbbb阅读 2,337评论 0 2
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,599评论 18 139
  • 原文链接date:20170710 Solidity中合约的概念和其他面向对象语言中的类差不多。他们都有状态变量来...
    gaoer1938阅读 969评论 0 0
  • 缅甸近几年森林损失速度居全球第三。有关专家警告称,若森林保护措施不当,缅甸境内森林将于2060年完全消失。 -中华...
    弘锦轩阅读 328评论 0 0
  • 在上海 学会了辨别方向 因为一条路很长 分东西南北; 也更精通于地图 去哪都不会让自己丢…… 一个人 只要有方向 ...
    沐16阅读 172评论 0 0