3步 老司机教你如何在以太坊上构建基于Token去中心化投票系统

  • 时间:
  • 浏览:47

  

  作者 | Doug Crescenzi

  译者 | 王柯凝

  出品 | CSDN、区块链大本营

  如果想在以太坊平台上构建一个去中心化的自治系统,其实有很多种不同的方法可供你选择。其中,最常用的方法之一就是,选民使用代币(Token)代表投票。你拥有的代币越多,就意味着你可以投的票数就越多。

  事实上,基于代币的去中心化投票系统有很多种类型,但基本上都要遵循一些固有的约定。通常来说,工作流程如下:

  是不是超级简单?

  然而,在构建一个基于代币的去中心化应用程序的实际操作中,我们还是会遇到很多困难与挑战。其中,最大的挑战之一就是选民欺诈行为。除此之外,投票的状态如何管理?一些代币独有的特性如何与相应的自治机制相结合?同样也是我们所要面临的挑战。

  那么,下面我们就来实操一下吧!

  现在,假设我们想要创建两个交互的智能合约:

  在这里,选民使用代币余额对提案进行投票,而代币余额则表示选民可以对某一特定提案投票的票数。

  首先,我们先假设某一选民提交提案。

  我们需要创建一个函数,允许选民向自治智能合约提交提案。

  代码如下:

  

  在允许选民提交提案之前,首先需要验证这些选民是否拥有投票资格。在我们这个案例中,选民的代币余额必须大于0。另外,我们使用onlyEligibleVoter修改器来检查选民是否拥有投票资格:

  

  一旦确定该选民拥有投票资格,submitProposal函数就会将该选民的代币余额传递给votesReceived变量,然后再将这个值作为参数传递给addProposal函数:

  

  如上所示,addProposal函数将为该提案生成一个唯一ID,并创建一个提案对象,用来监控提案收到的票数、提案是否已通过以及提案的投票截止日期。

  添加提案后,submitProposal函数将返回提案的唯一ID值,并触发ProposalSubmitted事件,供前端使用。

  第二种情况是,其他选民可以投赞成或反对票。

  现在,我们需要创建一个函数,来处理选民对该特定提案的投票。

  在这个函数里,首先使用一个if语句,判断选民是否被阻止投票:如果选民未被阻止投票,就需要确定选民的代币余额是多少,然后再将该余额值添加到该提案的votesReceived变量中。

  

  然后,我们就可以查看,经过投票,该提案是否已经通过了

  第三种情况是,投票通过并触发相应的动作。

  这并不是说我们已经成功创建了一个允许选民创建提案和提交选票的函数。下面,我们来看看,如果提案获得了足够多的票数后,会发生什么。

  submitVote函数将会调用if语句,检查是否已经收到足够的票数使提案通过:

  

  如果投票系统已经接收到足够多的票数,则会调用endVote函数:

  

  在这里,我们使用require语句和voteSuccessOrFail函数,确保该提案已经获得足够多的票数:

  

  如果投票通过,将会调用一个将Proposal的passed变量更新为True的函数。

  如何防止选民欺诈行为?

  在创建一个基于代币的去中心化投票应用程序的过程中,我们面临的最大挑战就是——选民的欺诈行为。比如,哪些因素会影响选民对提案进行投票?然后将代币发送给另外一个钱包地址进行重复投票?

  你还记不记得有一个名为blocked的映射?

  

  我们用该映射来检查选民是否被阻止投票:

  为了在我们的Token智能合约中实现这一点,我们将使用whenNotBlocked修改器:

  

  然后,我们就可以在智能合约中的transfer和transferFrom 函数中使用修改器,这两个函数继承于OpenZeppelin的ERC-20Token智能合约。

  

  当然,这种算法也有自身的缺点,即会抑制选民投票,因为如果选民处于blocked锁定状态,他们就无法转移代币。这篇文章也介绍了其他几种使用ERC-20 Token进行安全投票的方法。

  精华总结

  — END —