ERC-20代币是一种代币标准,由V神于2015年6月提出此标准,由Fabian Vogelsteller(erc20及erc725作者,也是LUKSO Token 创始人)于2015年11月提交EIP-20。该标准允许开发者在以太坊区块链网络上开发属于自己的代币,该标准定义了开发者可以按照标准要求使用一些简单的功能如: - 设定代币名称 - 设定代币总量 - 规定小数点位数 - 规范如何批准代币交易 - 如何访问数据 - 允许查看各地址中erc20代币数目以及代币总量 - 一定条件下,允许第三方账户使用某账户中的代币资产 - 允许代币和兼容ETH的智能合约及钱包服务等第三方个体兼容 以及一些简单的函数功能等等。# n# F3 G% c* l0 a F- a' X/ r6 s6 U7 H
4 O' E. p2 g% } W6 l. E
- pragma solidity >=0.7.0 <0.9.0;: ^. l+ G, M% M$ c. `3 P
-
- //投票实验
- contract Ballot{
- 5 ~" ~2 j7 q4 Y
- struct Voter{
- uint weight;//投票(单票)权重( Z) l* u. I) |: Q5 k- z
- bool voted;//是否投过票,true为投过票,bool类型默认值为false* F3 Z8 I/ Q) z) |. O0 C
- address delegate;//想要委托投票的节点地址,address默认值为0x0/ S% z9 I; d5 a) p V' o: r
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)$ t' M( z( y+ p* ?
- }) I; X. U' `1 u X
-
- struct Proposal{//被投票节点的相关参数
- bytes32 name;//被投票节点的姓名标识
- uint voteCount;//累积被投票数
- }
- Q K3 n% i! g8 i
- address public chairperson;//投票管理员地址5 K+ ]% M: o; D( g" r
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射
- Proposal[] public proposals;//一维数组存储被投票节点信息2 A! o4 u7 o/ R" u& v3 k
-
- //构造方法、构造函数
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数
- //且一个合约只能有一个构造函数,不能进行构造函数重载
- constructor(bytes32[] proposalNames) public{
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员. p7 c) x- j" B" i4 u' L4 x
- voters[chairperson].weight = 1;//将管理员投票权置为14 |; e" m' y( n& q) \- z
- ; Y4 h7 j8 H% B( {
- for(uint i=0; i<proposalNames.length; i++){6 s! }* f' [: |( R" D/ _9 }
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票5 q% b" m6 s! C: M1 L3 A
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));# w9 G# N# {5 r, r9 j
- proposals.push(Proposal({# |% i5 c8 ?$ S$ x
- name:proposalNames[i],' d0 i4 { t% p5 ~& P
- voteCount:0
- }));& k, H- s2 `$ k: |
- }
- }) e9 E, H, ]4 k7 k2 u8 G
-
- //由管理员授权可投票节点
- function giveRightToVote(address voter) public{( S7 H8 D% x. ]" `2 z
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,
- //未消耗的gas会被退回,合约状态会回退到初始状态
- require(# n9 F( q* K! h0 ^# z
- msg.sender == chairperson,"Only chairperson can give right to vote."; j4 i6 v* i) c; v/ p
- );//执行此function的节点一定为管理员节点
- require(4 [: |. ^9 r$ k a- M1 Y
- !voters[voter].voted,"The voter already voted."& r! \0 q, T2 T9 F5 E6 C
- );//若voter没投过票
- require(voters[voter].weight == 0);1 A# m- Q' [$ |
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权; `4 u3 b" C9 v" w# _
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数$ m k# P5 j4 _, e. {5 F3 z/ [, h
- }
- 1 J. H* d- I1 i$ s. G* G- m
- //投票授权
- 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.");
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己/ P g6 w% H0 ^. ^( l e5 ^
-
- //判断代理节点地址是否为空:address(0)或者address(0x0)
- while(voters[to].delegate != address(0)){
- to = voters[to].delegate;//找到最终的代理节点
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态
- }5 D, O; A" G1 r. }( w- H1 R' o
-
- sender.voted = true;//票权代理出去,状态改为已投票
- sender.delegate = to;//票权代理地址
- Voter storage delegate_ = voters[to];//取出代理节点状态
- * K# `- Q& D, b6 Z/ K
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和
- if(delegate_.voted){
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{
- delegate_.weight += sender.weight;- `$ y7 q% n& [$ j
- }
- }2 C& }3 b7 r% c/ m
-
- function vote(uint proposal) public{* _: m; @3 c: F' d U- ]# H% y
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息
- require(!sender.voted,"Already voted.");//若sender未投过票4 o7 F# H/ c* d9 N. }
- sender.voted = true;//更改投票状态为已投过票
- sender.vote = proposal;//保存已投票节点
- proposals[proposal].voteCount += sender.weight;//票权加和- D, x t% e) c% J
- }* q+ f# s. v% e0 F1 E
- - n4 X$ S- ~/ N- F( r7 A
- //返回票数最多的节点在一维数组proposals中的索引
- function winningProposal() public view returns(uint winningProposal_){
- uint winningVoteCount = 0;
. \# N3 u5 j) ~5 o$ E
: j! s: J! d: T+ P* W' S
solidity投票智能合约代码,亲测可用, l ?& ]! w, T+ G; Z