+ F: d( k4 f+ H. p) H! q% X
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;* ?' b8 X5 G7 D2 I% g
-
- //投票实验
- contract Ballot{
-
- struct Voter{6 J5 y& C5 i) s$ x! x* N7 M0 M7 ?
- uint weight;//投票(单票)权重
- bool voted;//是否投过票,true为投过票,bool类型默认值为false
- address delegate;//想要委托投票的节点地址,address默认值为0x0
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)
- }" B5 [! r/ M7 D' w
- + j! `* P# b# q
- struct Proposal{//被投票节点的相关参数
- bytes32 name;//被投票节点的姓名标识4 V5 Z: |& l2 O% F8 l9 c
- uint voteCount;//累积被投票数
- }; L7 a/ V, f% Z" f
-
- address public chairperson;//投票管理员地址
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射2 Q0 G8 y! L- d% o; Z
- Proposal[] public proposals;//一维数组存储被投票节点信息
-
- //构造方法、构造函数
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数
- //且一个合约只能有一个构造函数,不能进行构造函数重载
- constructor(bytes32[] proposalNames) public{
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员
- voters[chairperson].weight = 1;//将管理员投票权置为13 [& I* ^4 z: V& H# ^, V W; Z
- 3 x$ W1 b( _, `0 z6 C. l5 v
- for(uint i=0; i<proposalNames.length; i++){
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票: S% W- a, [- m+ {
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));
- proposals.push(Proposal({
- name:proposalNames[i],
- voteCount:0- R& k' Y, e0 J- `
- }));8 z+ H, A! i" K' q/ {: e v
- }
- }0 K3 ~) N1 }& O( P5 |6 h
- 1 b7 q0 P ~" {0 a! u2 U4 Z
- //由管理员授权可投票节点' s* F5 p+ m- m: i0 Z8 R7 w3 I% W. [
- function giveRightToVote(address voter) public{1 \, k/ ?; K' ~2 [0 X; C
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,1 @4 s4 E1 U& N0 V% V+ x3 \
- //未消耗的gas会被退回,合约状态会回退到初始状态
- require() t" s9 u2 C" u. w
- msg.sender == chairperson,"Only chairperson can give right to vote."+ M2 @. y2 C) M. M0 U
- );//执行此function的节点一定为管理员节点
- require(+ r) I b- Y' U% e) f9 ]
- !voters[voter].voted,"The voter already voted."
- );//若voter没投过票 ?& x- |; G6 m& M
- require(voters[voter].weight == 0);
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数
- }
- ) r( |. W7 [; \: u4 }
- //投票授权6 |8 {; W9 T- y' m1 I
- function delegate(address to) public{
- Voter storage sender = voters[msg.sender];, ]6 E5 N9 y. y4 S
- require(!sender.voted, "You already voted.");
- require(to != msg.sender,"Self-delegation is disallowed.");
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己9 a' m6 N% d. p. u* T4 |
- ( b @( H5 j( g+ }
- //判断代理节点地址是否为空:address(0)或者address(0x0)& W, K+ @1 ?, J5 r9 ]& L" A/ ^
- while(voters[to].delegate != address(0)){2 @" q3 z7 z0 f9 _! a: n
- to = voters[to].delegate;//找到最终的代理节点& X* O" ^& Y- {, M
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态+ O: Z1 T. G! [! _# m
- }: J- U) H! t# c9 K
-
- sender.voted = true;//票权代理出去,状态改为已投票
- sender.delegate = to;//票权代理地址
- Voter storage delegate_ = voters[to];//取出代理节点状态, c# z `- f' E( c5 h
- / N' | ~# v T! }
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和0 l& j1 j$ z% P! a& F, n! w9 ^' j
- if(delegate_.voted){' W5 f/ z1 \ U. j1 e
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{
- delegate_.weight += sender.weight;
- }
- }
-
- function vote(uint proposal) public{
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息! w1 R; ~& q! A. I& w B6 o
- require(!sender.voted,"Already voted.");//若sender未投过票
- sender.voted = true;//更改投票状态为已投过票7 E3 _" l% r+ q. w4 j& X) H, o
- sender.vote = proposal;//保存已投票节点' V; _; M* N2 }
- proposals[proposal].voteCount += sender.weight;//票权加和
- }1 J7 S+ Y; @: i6 l
- 2 W x$ |, Z, V9 }& I" @
- //返回票数最多的节点在一维数组proposals中的索引8 ]# ^7 U6 r2 Z* Y( i
- function winningProposal() public view returns(uint winningProposal_){
- uint winningVoteCount = 0;
- {# b5 i% x0 W4 R
7 ^& L0 V. W4 |$ w4 S2 n
solidity投票智能合约代码,亲测可用9 P D0 E0 A' U