ERC-20代币是一种代币标准,由V神于2015年6月提出此标准,由Fabian Vogelsteller(erc20及erc725作者,也是LUKSO Token 创始人)于2015年11月提交EIP-20。该标准允许开发者在以太坊区块链网络上开发属于自己的代币,该标准定义了开发者可以按照标准要求使用一些简单的功能如: - 设定代币名称 - 设定代币总量 - 规定小数点位数 - 规范如何批准代币交易 - 如何访问数据 - 允许查看各地址中erc20代币数目以及代币总量 - 一定条件下,允许第三方账户使用某账户中的代币资产 - 允许代币和兼容ETH的智能合约及钱包服务等第三方个体兼容 以及一些简单的函数功能等等。3 d. j z n" N1 D' }! D
6 w% |7 F$ p: M6 ~
- pragma solidity >=0.7.0 <0.9.0;
- 6 g" o7 F$ A" y7 |& }; l
- //投票实验
- contract Ballot{
- , E0 F* I( d: h! y# g e1 n
- struct Voter{/ { g% z4 [8 a9 Y$ v3 z
- uint weight;//投票(单票)权重
- bool voted;//是否投过票,true为投过票,bool类型默认值为false
- address delegate;//想要委托投票的节点地址,address默认值为0x0! j4 u' H" a: u% B5 S# M0 o% i
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)
- }( {4 d" S, J% J1 n; g; t9 }
-
- struct Proposal{//被投票节点的相关参数4 S$ N3 _0 C7 {+ y5 v" @! U4 F
- bytes32 name;//被投票节点的姓名标识
- uint voteCount;//累积被投票数' h# T' M8 f2 W' |1 t4 @0 w% e
- }2 P( i; B# S+ M2 e
- 1 v6 A( a; ~9 m/ j
- address public chairperson;//投票管理员地址$ J: A1 x7 E0 p& ?7 K3 i
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射
- Proposal[] public proposals;//一维数组存储被投票节点信息) v$ p0 l/ J R) Q
-
- //构造方法、构造函数
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数
- //且一个合约只能有一个构造函数,不能进行构造函数重载
- constructor(bytes32[] proposalNames) public{& ~3 l( O. t2 Y: d8 z7 C
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员
- voters[chairperson].weight = 1;//将管理员投票权置为10 q" K: H5 l9 M+ }1 [ o2 y7 h. ]. ]
-
- for(uint i=0; i<proposalNames.length; i++){
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));2 N# a8 n1 U* m3 g/ D; t# G# b8 l
- proposals.push(Proposal({( ]4 T8 |. U5 x- U9 W" f
- name:proposalNames[i],
- voteCount:0
- }));
- }: O- o/ V* r+ t1 s/ P5 ~
- }
- 1 w7 v5 W. T- v9 i# E7 \7 d. U
- //由管理员授权可投票节点; X$ m, y5 Y, s! S, r3 f7 E
- function giveRightToVote(address voter) public{
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,/ B; }8 u! z0 B+ S* |
- //未消耗的gas会被退回,合约状态会回退到初始状态' b2 G9 N0 K2 @: k/ Z- p: t0 F
- require(
- msg.sender == chairperson,"Only chairperson can give right to vote."0 B& d+ Y( a7 \
- );//执行此function的节点一定为管理员节点
- require(
- !voters[voter].voted,"The voter already voted.": l) J; ?' Q. M
- );//若voter没投过票! V6 m/ E* q" }# _# t; N
- require(voters[voter].weight == 0);
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数0 R' }3 S: F3 J
- } t, v, t! e+ |
-
- //投票授权
- function delegate(address to) public{% c: c/ z1 B p E0 ~
- Voter storage sender = voters[msg.sender];
- require(!sender.voted, "You already voted.");
- require(to != msg.sender,"Self-delegation is disallowed.");: Z) Q& ~4 C+ z5 ?
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己
- , \2 d+ S: c C o& K/ R4 c& S: Z
- //判断代理节点地址是否为空:address(0)或者address(0x0)
- while(voters[to].delegate != address(0)){' P3 [* @ v0 S# O; u1 z( {3 `; X
- to = voters[to].delegate;//找到最终的代理节点
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态! J: K9 \8 W, l: Y3 P7 y' Y, i/ X2 h
- }- j% x* m& b' `6 j0 u
-
- sender.voted = true;//票权代理出去,状态改为已投票/ d+ o& G9 D- q0 H) V' P. ]! f
- sender.delegate = to;//票权代理地址
- Voter storage delegate_ = voters[to];//取出代理节点状态
- . i! z% E0 A6 T1 e/ o, T( C& i
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和5 S# d0 p# N- ~
- if(delegate_.voted){5 p; t3 R7 t# f) c4 C
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{
- delegate_.weight += sender.weight;
- }
- }& v- P% C. _5 i$ R
- " Q8 A9 A, k# C$ F0 V% P! Q
- function vote(uint proposal) public{' P6 g2 k& @3 N5 P
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息
- require(!sender.voted,"Already voted.");//若sender未投过票2 {# h/ y& U, w2 K1 L
- sender.voted = true;//更改投票状态为已投过票
- sender.vote = proposal;//保存已投票节点/ `9 ?% ~* s# K2 Q1 {; G. i& \
- proposals[proposal].voteCount += sender.weight;//票权加和
- }
- 3 P3 a. K @; Q- {3 W
- //返回票数最多的节点在一维数组proposals中的索引) u4 q* E0 i8 q5 ?, E$ P# }
- function winningProposal() public view returns(uint winningProposal_){7 u1 X+ O& V/ t
- uint winningVoteCount = 0;: ~7 b) \2 d! M3 `4 r( ]% p/ l
% R/ H1 Q x8 a0 d
2 }4 L4 ]& [7 S4 V! y
solidity投票智能合约代码,亲测可用! ~# Z) F& I: T7 B; L# q3 P0 Z