Hi 游客

更多精彩,请登录!

比特池塘 区块链技术 正文

solidity投票智能合约代码

BlockQ
144 0 0
    solidity投票智能合约代码,电子投票功能要解决的主要问题是如果分配投票权以及如何避免数据被篡改。本篇实现的合约思路是对于每次投票表决都创建一个合约,合约的创建者就是投票委员会的主席,可以给不同的账户投票的权利。拥有投票权的账户可以自己投票也可以委托给他所信任的人代理投票。$ }, e* l) E5 N1 R* B

$ I$ @2 e( ?' J+ J  W7 o
  1. pragma solidity >=0.7.0 <0.9.0;
      _& W6 x8 O% y& V2 O
  2. & u: N! X* O7 }. Z" [% w6 S
  3. //投票实验& @1 u3 N6 U0 l3 h2 y- x
  4. contract Ballot{* V7 T5 L2 ]2 C
  5.     struct Voter{/ c3 `/ n+ {# x* r/ B+ U) W
  6.         uint weight;//投票(单票)权重/ D8 E- p3 h9 b5 h. j& ^! Q+ F
  7.         bool voted;//是否投过票,true为投过票,bool类型默认值为false
    1 I* Z  F0 ]  F$ k7 m0 v
  8.         address delegate;//想要委托投票的节点地址,address默认值为0x0
      p7 K5 G8 ^6 M' F. V
  9.         uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)
    , a. P5 g9 V2 S' X
  10.     }2 v) a/ _1 z9 }" {2 J
  11.      3 `& O2 ]; m4 r* ^
  12.     struct Proposal{//被投票节点的相关参数
    7 h; E3 E* F/ r7 ]
  13.         bytes32 name;//被投票节点的姓名标识3 P% u) c( p7 r2 |) o. [0 }
  14.         uint voteCount;//累积被投票数
    : h, b0 u( H+ z; x6 c$ N
  15.     }- S  K( A" `. V  h* I
  16. 9 i) @8 \6 H9 x3 E7 W! v$ J4 S
  17.     address public chairperson;//投票管理员地址4 c! W! e2 N! d7 O  ~* w
  18.     mapping(address => Voter) public voters;//地址对投票节点的信息的映射
    / m; a7 w$ Q; o5 X
  19.     Proposal[] public proposals;//一维数组存储被投票节点信息
    - r" M  q; K5 O. J/ s: F& B

  20. 1 Z' f! M5 B6 y0 R( p
  21.     //构造方法、构造函数
    4 [9 W; M* F/ X1 d
  22.     //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数2 _; n3 k2 [& U- ]3 [, ^! C$ ~
  23.     //且一个合约只能有一个构造函数,不能进行构造函数重载
    . e- Y$ H5 C7 F8 J  P2 h% b0 G# g
  24.     constructor(bytes32[] proposalNames) public{
    5 w+ Q% f' S$ Q* ^
  25.         chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员. F% |, [0 A- x# h9 o! D$ ]7 ^7 s
  26.         voters[chairperson].weight = 1;//将管理员投票权置为1
    6 a: B2 d1 E/ _: _9 @) |6 E
  27. 7 o% g8 P) P& [" A
  28.         for(uint i=0; i<proposalNames.length; i++){
    5 c8 c" h( y1 k1 u$ ~
  29.             //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票3 w6 G9 Z6 y% }9 X' o+ y
  30.             //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));
    7 m" Z3 M2 g8 D9 T8 |! v# H; p
  31.             proposals.push(Proposal({
      G  W$ k1 V/ a6 ~4 L$ c$ e
  32.                 name:proposalNames[i],
    , k8 n+ S; b- d
  33.                 voteCount:0  A* p6 Q; u, ~7 L
  34.             }));& W& o8 {. W+ i
  35.         }$ \" r" f+ r# R0 P) s
  36.     }" T, u: O7 t" l9 f2 C

  37. ; ?' W0 y' k* U5 p
  38.     //由管理员授权可投票节点
    % L6 |' z. z: v
  39.     function giveRightToVote(address voter) public{5 d2 Y8 U9 |! y% b; j
  40.         //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,. q0 }& x7 \3 ?4 x
  41.         //未消耗的gas会被退回,合约状态会回退到初始状态% r! r9 j4 W; q6 l
  42.         require(# K  Y% v" h! N. Z' Q( F
  43.             msg.sender == chairperson,"Only chairperson can give right to vote."  {% |5 D1 d+ I" |3 y; ]* G0 q4 `
  44.         );//执行此function的节点一定为管理员节点3 c+ r3 ^& ^. `# ?0 x( \; r
  45.         require(  J$ Q9 D7 P6 y; ]6 ~) j- [
  46.             !voters[voter].voted,"The voter already voted."; u' {  L6 V8 X1 i% I) [
  47.         );//若voter没投过票: u) R( ^+ F$ B) l
  48.         require(voters[voter].weight == 0);
    # j7 y8 |/ |' N- v: R
  49.         //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权- L% h2 R7 Q. d# ]' a
  50.         voters[voter].weight = 1;//上述三个require()均成立时,授权票数9 v1 F5 t4 }: X7 j
  51.     }" v# [( |/ J* [

  52. % d) V' m% U5 J$ G4 ^6 L
  53.     //投票授权  p: r- p* V" A+ z
  54.     function delegate(address to) public{) l* I0 e* a; p
  55.         Voter storage sender = voters[msg.sender];
    + j% x7 g. }6 J
  56.         require(!sender.voted, "You already voted.");
    7 R. X  K, V! _4 a( e9 _
  57.         require(to != msg.sender,"Self-delegation is disallowed.");0 O3 B  Z  I: j; C  p* Y' w
  58.         //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己! x- V6 c$ G7 W3 l3 `
  59. + P# t0 Y6 n# k  Y  o  Q7 y8 B
  60.         //判断代理节点地址是否为空:address(0)或者address(0x0)8 z0 p# i  {6 x' F$ R, o
  61.         while(voters[to].delegate != address(0)){' [" L% f% P3 s: a8 h7 G
  62.             to = voters[to].delegate;//找到最终的代理节点5 v+ F; `5 S1 {+ q6 ?% X) d0 f- l  B! u
  63.             require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态  {8 ?* Q  g* B8 u
  64.         }
    4 k9 O) A/ [; Z( c+ O6 Y
  65. 1 L! _( F9 k& N% I2 e, X$ i. j/ W" A
  66.         sender.voted = true;//票权代理出去,状态改为已投票
    ' G+ p. n5 A) `$ e5 ?/ W: z
  67.         sender.delegate = to;//票权代理地址# `( g+ c# \+ E3 c' z: `
  68.         Voter storage delegate_ = voters[to];//取出代理节点状态
    # a" V: Z& B; n1 F2 e
  69.   S+ S2 v/ e+ W# ?( F
  70.         //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和
    6 d2 G  q9 X6 X) y
  71.         if(delegate_.voted){$ [; z+ m+ V" z2 X/ p
  72.             proposals[delegate_.vote].voteCount += sender.weight;
    ' I' t2 J( g1 C# \+ ~7 Z. b& [
  73.         }else{
    & [( M- V+ U: ~- a0 r' T
  74.             delegate_.weight += sender.weight;( ^, I! f: v7 e9 G( x4 K
  75.         }
    % e7 T$ q0 b. M8 Z$ M
  76.     }
    6 ]9 C$ X% }6 q

  77. . G- E  W4 p6 B) w
  78.     function vote(uint proposal) public{, m1 [% K' v* H0 b$ [# a
  79.         Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息; @, D1 U0 q+ o+ g: }3 t) X% A* e
  80.         require(!sender.voted,"Already voted.");//若sender未投过票, G. a/ D7 w" {9 @
  81.         sender.voted = true;//更改投票状态为已投过票+ l4 U- o3 Y3 ~+ G' L$ }7 p. h+ M
  82.         sender.vote = proposal;//保存已投票节点  f& W+ Q$ T7 t/ ~" A. q
  83.         proposals[proposal].voteCount += sender.weight;//票权加和
    7 A: G7 W/ l1 V6 J* ~
  84.     }
    % a* ]8 k& j, P9 b1 W
  85. / f: g/ p+ D/ d7 G! p
  86.     //返回票数最多的节点在一维数组proposals中的索引
    1 j+ c, i. A( Y. @: F7 G" p
  87.     function winningProposal() public view returns(uint winningProposal_){
    8 y; i- S! @1 s2 w: A
  88.         uint winningVoteCount = 0;
    * T4 v+ \) \4 A; @
  89.         for(uint p=0;p<proposals.length;p++){
    ; i% u; l. d; C
  90.             if(proposals[p].voteCount > winningVoteCount){, J8 r2 S  R+ b( q2 N  o! j
  91.                 winningVoteCount = proposals[p].voteCount;  h9 e# `4 |8 O
  92.                 winningProposal_ = p;5 w9 t. t* u# P% I
  93.             }
    1 Z$ B* s; M& Y2 j" K' F/ c* B
  94.         }0 B8 m4 U+ A: l% I8 }: G7 {. D! M
  95.     }* F7 M, o& O: H( Q- u
  96.     //输出票数最多的节点name
    : T3 f+ Q* h8 |
  97.     function winnerName() public view returns(bytes32 winnerName_){
    $ G5 D. P5 h7 d' e& y4 W/ D2 C
  98.         winnerName_ = proposals[winningProposal()].name;. y8 S$ p7 D0 p' H6 z8 b; S. I
  99.     }
    2 O# T2 O4 e0 ]. u
  100. }
复制代码

; d2 K8 d8 J" p$ z. d! |) X, X/ K3 U
BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

成为第一个吐槽的人

BlockQ 小学生
  • 粉丝

    0

  • 关注

    0

  • 主题

    3