9 q# H& G6 B! h* \4 V5 N
ERC-20代币是一种代币标准,由V神于2015年6月提出此标准,由Fabian Vogelsteller(erc20及erc725作者,也是LUKSO Token 创始人)于2015年11月提交EIP-20。该标准允许开发者在以太坊区块链网络上开发属于自己的代币,该标准定义了开发者可以按照标准要求使用一些简单的功能如: - 设定代币名称 - 设定代币总量 - 规定小数点位数 - 规范如何批准代币交易 - 如何访问数据 - 允许查看各地址中erc20代币数目以及代币总量 - 一定条件下,允许第三方账户使用某账户中的代币资产 - 允许代币和兼容ETH的智能合约及钱包服务等第三方个体兼容 以及一些简单的函数功能等等。
- pragma solidity >=0.7.0 <0.9.0;
-
- //投票实验+ V: Y+ c. K, c
- contract Ballot{
-
- struct Voter{" C$ N* q% Q5 K- B5 v. u9 F8 x
- uint weight;//投票(单票)权重5 A7 p' ~1 b5 Z6 r' O( {1 P3 b% w7 I
- bool voted;//是否投过票,true为投过票,bool类型默认值为false
- address delegate;//想要委托投票的节点地址,address默认值为0x0
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)
- }9 K) h4 h+ F" q R2 e0 l* | P
- 0 `8 w' n1 s, {. f
- struct Proposal{//被投票节点的相关参数" N& ? b. ?- H, f# M$ |
- bytes32 name;//被投票节点的姓名标识 I1 h4 @5 f9 r
- uint voteCount;//累积被投票数" b3 ^- u+ }) ^ Z' @4 v6 D
- }, |' H8 w9 z" D$ \$ O4 ^
-
- address public chairperson;//投票管理员地址
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射% e' ]) C2 m8 Y# n/ x: X; K4 f
- Proposal[] public proposals;//一维数组存储被投票节点信息
-
- //构造方法、构造函数, ?- ~, w% u* L) O
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数4 X6 E* H7 ^' v8 Y
- //且一个合约只能有一个构造函数,不能进行构造函数重载- b# d0 N5 O; @9 }! Q3 H
- constructor(bytes32[] proposalNames) public{
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员
- voters[chairperson].weight = 1;//将管理员投票权置为13 S3 j3 c+ i/ y" ]- @
-
- for(uint i=0; i<proposalNames.length; i++){4 g% I% X% }# G* C/ J
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票 |. i3 h) A7 K$ w
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));, o" d- e: i2 v/ m3 f% i
- proposals.push(Proposal({# K) r. |9 I$ g [* y2 z
- name:proposalNames[i],: J9 y9 Q; g5 n. W
- voteCount:0
- }));
- }: f; W5 `2 B, v
- }
- 1 ^8 v# N. H: x+ ^9 F6 h3 ?
- //由管理员授权可投票节点3 j# J8 I* F. L" h
- function giveRightToVote(address voter) public{7 f3 Q% _7 ?/ y4 j- N6 ?, V4 J
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,4 P! p4 X9 x( m2 y+ v4 N# @9 ^
- //未消耗的gas会被退回,合约状态会回退到初始状态- d% J0 Q) l/ L; U5 z
- require(/ ?* F; k7 I5 A2 b) r) T# h% X
- msg.sender == chairperson,"Only chairperson can give right to vote."
- );//执行此function的节点一定为管理员节点, e4 `: r" C+ d8 u* _0 f7 ?
- require(
- !voters[voter].voted,"The voter already voted."5 {; a3 |& X. ^' T
- );//若voter没投过票
- require(voters[voter].weight == 0);! B* ~4 _4 ?- j7 l p
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数0 J' R) y) Z+ S
- }
- * T5 V5 }& b; A, u* t
- //投票授权/ `& Z1 R1 U0 ^6 o5 e$ ^& M
- function delegate(address to) public{; H9 J+ ?, c1 C. N$ y
- Voter storage sender = voters[msg.sender];
- require(!sender.voted, "You already voted.");) o' u/ v2 e6 d
- require(to != msg.sender,"Self-delegation is disallowed.");
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己( o5 m4 r: P" L+ r
-
- //判断代理节点地址是否为空:address(0)或者address(0x0)0 l# U( r6 M9 h r7 q- v7 o* r
- while(voters[to].delegate != address(0)){/ F( f7 b) t. Q m2 ^9 ?
- to = voters[to].delegate;//找到最终的代理节点& h, h; a ~/ b* i
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态1 @' `* _& b. Q& _8 V7 ~
- } U) T: o; k) a1 a4 V7 X. Q0 m
-
- sender.voted = true;//票权代理出去,状态改为已投票
- sender.delegate = to;//票权代理地址* g& r. B: R. d+ {' M
- Voter storage delegate_ = voters[to];//取出代理节点状态
-
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和' e7 }+ f8 l" {0 X& w- {
- if(delegate_.voted){
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{
- delegate_.weight += sender.weight;
- }
- } V# o, X( N( X
- 6 K) [! H9 k) O( i# F
- function vote(uint proposal) public{
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息8 a, X& G6 B1 _
- require(!sender.voted,"Already voted.");//若sender未投过票
- sender.voted = true;//更改投票状态为已投过票, {# F W4 z) @7 C/ k, C5 M0 E- ~
- sender.vote = proposal;//保存已投票节点
- proposals[proposal].voteCount += sender.weight;//票权加和! n# E3 `3 K- F- g; C
- }
-
- //返回票数最多的节点在一维数组proposals中的索引
- function winningProposal() public view returns(uint winningProposal_){* U4 y% r* c) `; }( y, c
- uint winningVoteCount = 0;
) s5 A# e$ {3 y2 f t9 l: u+ y
0 y: t5 ]. I! i, V5 M, C
solidity投票智能合约代码,亲测可用