ERC-20代币是一种代币标准,由V神于2015年6月提出此标准,由Fabian Vogelsteller(erc20及erc725作者,也是LUKSO Token 创始人)于2015年11月提交EIP-20。该标准允许开发者在以太坊区块链网络上开发属于自己的代币,该标准定义了开发者可以按照标准要求使用一些简单的功能如: - 设定代币名称 - 设定代币总量 - 规定小数点位数 - 规范如何批准代币交易 - 如何访问数据 - 允许查看各地址中erc20代币数目以及代币总量 - 一定条件下,允许第三方账户使用某账户中的代币资产 - 允许代币和兼容ETH的智能合约及钱包服务等第三方个体兼容 以及一些简单的函数功能等等。
! S4 E8 E! y2 I1 L0 d4 C
- pragma solidity >=0.7.0 <0.9.0;
- ) Q. {$ x2 d* u( `. _4 B
- //投票实验
- contract Ballot{! B% j( i. k6 e
- ( I) Y0 I* A% O& g+ g
- struct Voter{/ A' C/ M/ A# e8 L/ z" \3 s
- uint weight;//投票(单票)权重. g7 _3 t+ j/ M4 O) S
- bool voted;//是否投过票,true为投过票,bool类型默认值为false
- address delegate;//想要委托投票的节点地址,address默认值为0x08 S0 [3 o# P# [& u- M
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)8 F: k* k+ O& k9 G* F! j1 x
- }
-
- struct Proposal{//被投票节点的相关参数8 T/ m' E q# Z$ J7 M" J3 r d% H) S
- bytes32 name;//被投票节点的姓名标识
- uint voteCount;//累积被投票数4 i* C0 L# p* D. W4 s
- }. S/ x9 r$ |: _. F" T+ f9 q
- ) C1 w6 |" {) N- \7 q+ `
- address public chairperson;//投票管理员地址3 n. r. I M7 C' i' Z' A
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射
- Proposal[] public proposals;//一维数组存储被投票节点信息
-
- //构造方法、构造函数
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数" u3 ^! E" T* _* g* p
- //且一个合约只能有一个构造函数,不能进行构造函数重载
- constructor(bytes32[] proposalNames) public{
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员
- voters[chairperson].weight = 1;//将管理员投票权置为1, s7 W7 ?5 U2 I0 N& e M
-
- for(uint i=0; i<proposalNames.length; i++){! x. K6 u8 @2 d- o V" w' T+ r
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票, Y+ B4 [+ ]- S* h% g2 P
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));0 H+ e1 L+ H5 l1 w Y2 L
- proposals.push(Proposal({
- name:proposalNames[i],
- voteCount:0 J4 n( I$ y, j( @* P0 R1 m
- }));
- }
- }2 G1 }) B. [3 j5 i1 E+ H
-
- //由管理员授权可投票节点
- function giveRightToVote(address voter) public{
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,1 T1 i5 D5 |% s' E! }
- //未消耗的gas会被退回,合约状态会回退到初始状态; m: j) O3 {7 i3 t
- require($ ?4 ]. x( N9 v
- msg.sender == chairperson,"Only chairperson can give right to vote."! J* i% |( d p7 c& a
- );//执行此function的节点一定为管理员节点* T5 b! A" w* k; Q) a
- require(
- !voters[voter].voted,"The voter already voted."; e" f' W: [6 ]# _7 p+ e& L% L2 w
- );//若voter没投过票1 {# |$ i' H! Y" ~
- require(voters[voter].weight == 0);+ T3 V6 y) x) v
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数( ~ R! c# }& V l) |
- }
- 5 r: o, I1 N6 s. S; s, V
- //投票授权" `5 }6 P. `- @7 C, a! D' Y
- function delegate(address to) public{
- Voter storage sender = voters[msg.sender];
- require(!sender.voted, "You already voted.");
- require(to != msg.sender,"Self-delegation is disallowed.");& [3 [- D1 S" A3 V
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己
-
- //判断代理节点地址是否为空:address(0)或者address(0x0)
- while(voters[to].delegate != address(0)){
- to = voters[to].delegate;//找到最终的代理节点
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态
- }
- . Z$ q4 S, n# {3 P4 c
- sender.voted = true;//票权代理出去,状态改为已投票+ K& B" z6 P P. s
- sender.delegate = to;//票权代理地址5 G8 l7 P+ F0 m/ M9 K
- Voter storage delegate_ = voters[to];//取出代理节点状态
- 2 p4 W! W! n8 O
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和
- if(delegate_.voted){
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{1 Q: L! L) x% w* s& _# {2 ]4 Y! r
- delegate_.weight += sender.weight;
- }
- }+ P) F0 X M- d
-
- function vote(uint proposal) public{" p( q: T! [5 x4 ^ v
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息3 B0 O6 t! X4 h
- require(!sender.voted,"Already voted.");//若sender未投过票1 k$ e: `% Z" n+ X/ c @! M2 k5 q' r+ ~
- sender.voted = true;//更改投票状态为已投过票
- sender.vote = proposal;//保存已投票节点
- proposals[proposal].voteCount += sender.weight;//票权加和3 ?3 O8 N3 X( c4 N2 I, _
- }6 I9 Y7 d" _% \# u
-
- //返回票数最多的节点在一维数组proposals中的索引
- function winningProposal() public view returns(uint winningProposal_){/ d1 i3 |6 f# }2 { [" g
- uint winningVoteCount = 0;& g/ T4 g/ V- v) I6 t( V1 |% O! s
* |6 _+ v8 G" ~$ z4 p
- X, @$ Z% P' j. @; G
solidity投票智能合约代码,亲测可用/ k5 k# _! j4 t2 A0 l4 W. Z) @