- pragma solidity >=0.7.0 <0.9.0;$ Z# H: l6 |' V2 [" B. z/ |. e8 |
- 1 G. W8 p6 V0 J0 V7 O6 f3 k
- //投票实验
- contract Ballot{- t+ J: ?* N8 a) m. r, B; i
- struct Voter{8 Y2 L6 ^6 \+ ~
- uint weight;//投票(单票)权重8 A/ q) e# ?4 [3 {' R3 ^! ] u8 [
- bool voted;//是否投过票,true为投过票,bool类型默认值为false0 `; O! R( z- j. O( e
- address delegate;//想要委托投票的节点地址,address默认值为0x0
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)% B& ~* `( r5 K9 U
- }2 X2 |& E0 U' ]8 t
- , v. Q% T2 i8 [+ B; h! \
- struct Proposal{//被投票节点的相关参数0 o9 |+ s: u- e, D
- bytes32 name;//被投票节点的姓名标识
- uint voteCount;//累积被投票数4 t# a0 q& X% t8 g
- }
-
- address public chairperson;//投票管理员地址
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射
- Proposal[] public proposals;//一维数组存储被投票节点信息
-
- //构造方法、构造函数
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数
- //且一个合约只能有一个构造函数,不能进行构造函数重载
- constructor(bytes32[] proposalNames) public{* W9 H% i* G x3 r2 ]8 d* G7 }
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员; `+ H( `- Q( q$ D+ K8 I+ v, {
- voters[chairperson].weight = 1;//将管理员投票权置为18 i7 A& t) S6 g, v% v* h
-
- for(uint i=0; i<proposalNames.length; i++){6 s) }8 H3 W! L/ @5 L! q
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));+ ~. R+ N8 s0 @
- proposals.push(Proposal({: \* [$ r' U) s( j8 G5 r! c, `; r
- name:proposalNames[i],9 s5 v4 k8 _, y3 @9 Z- i3 c M
- voteCount:0$ Q, Y8 |# q4 M8 X D9 f, A5 k
- }));2 q& e# l: L% x
- }
- }
-
- //由管理员授权可投票节点& @4 Y3 R3 V% p' d$ d7 \: L; S
- function giveRightToVote(address voter) public{3 {- S1 ^& I; [
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,! G6 b% I# J# p' e
- //未消耗的gas会被退回,合约状态会回退到初始状态
- require(
- msg.sender == chairperson,"Only chairperson can give right to vote."
- );//执行此function的节点一定为管理员节点
- require(& g9 v: W4 r5 y' f0 z% s% F* t
- !voters[voter].voted,"The voter already voted."8 K9 S; Y' Y( `* Y; ~
- );//若voter没投过票
- require(voters[voter].weight == 0);! p) ]# c2 z6 s% F+ j! L* x0 k- I; o
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数$ b' J5 g8 N$ w* h% X
- }% Y- Q# v3 H- v2 ]6 \ a
-
- //投票授权5 B- t3 c! j* \3 ?
- function delegate(address to) public{
- Voter storage sender = voters[msg.sender];3 s" t8 S/ p$ p* k1 o- [
- require(!sender.voted, "You already voted.");: o, U% \ G4 z/ `0 r
- require(to != msg.sender,"Self-delegation is disallowed.");# ~3 q9 `) o0 ]4 w. u, ?2 \1 M
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己
- ! h6 h i4 U( V4 }5 @/ S( O
- //判断代理节点地址是否为空:address(0)或者address(0x0)* c6 E+ u7 g# S( _: x+ ]
- while(voters[to].delegate != address(0)){
- to = voters[to].delegate;//找到最终的代理节点
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态( R* j5 S4 v s+ G
- }1 C8 p; G3 x2 l! T2 K' x& j
- 2 e- Z: V: g6 C* b( N9 l1 `
- sender.voted = true;//票权代理出去,状态改为已投票4 _5 k5 F5 w1 t% n! A* N6 g l
- sender.delegate = to;//票权代理地址
- Voter storage delegate_ = voters[to];//取出代理节点状态
-
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和
- if(delegate_.voted){
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{4 D/ V$ C' n0 [+ _: H
- delegate_.weight += sender.weight;
- }
- }* N7 W2 Y9 {* j, j7 c
- 9 Q; j8 s' y1 ]6 o
- function vote(uint proposal) public{; b9 `5 R! T# h J! x% @
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息
- require(!sender.voted,"Already voted.");//若sender未投过票
- sender.voted = true;//更改投票状态为已投过票, b0 p3 ~8 H3 W0 Q2 H. }, ]
- sender.vote = proposal;//保存已投票节点- y( o# i/ u+ q. N- K/ s# y
- proposals[proposal].voteCount += sender.weight;//票权加和
- }" v$ w: [5 @3 N: u* h
-
- //返回票数最多的节点在一维数组proposals中的索引. `% v( E% S2 S: J; M8 n" F- e
- function winningProposal() public view returns(uint winningProposal_){
- uint winningVoteCount = 0;
- for(uint p=0;p<proposals.length;p++){
- if(proposals[p].voteCount > winningVoteCount){9 _9 c: D% Y5 a$ z/ B$ L
- winningVoteCount = proposals[p].voteCount;. K6 ^4 L/ Z" g
- winningProposal_ = p;, @$ M/ l ~1 N5 U% A
- } Y* ?3 v2 P2 y4 N! w6 Y' `
- }
- }
- //输出票数最多的节点name
- function winnerName() public view returns(bytes32 winnerName_){! {9 b6 l# |% l$ I9 \) Q
- winnerName_ = proposals[winningProposal()].name;
- }
- }
solidity投票智能合约代码
BlockQ
发表于 2022-11-11 20:33:36
140
0
0
BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
成为第一个吐槽的人