" x( b8 {3 v* s+ b# n, S8 l4 d
ERC-20代币是一种代币标准,由V神于2015年6月提出此标准,由Fabian Vogelsteller(erc20及erc725作者,也是LUKSO Token 创始人)于2015年11月提交EIP-20。该标准允许开发者在以太坊区块链网络上开发属于自己的代币,该标准定义了开发者可以按照标准要求使用一些简单的功能如: - 设定代币名称 - 设定代币总量 - 规定小数点位数 - 规范如何批准代币交易 - 如何访问数据 - 允许查看各地址中erc20代币数目以及代币总量 - 一定条件下,允许第三方账户使用某账户中的代币资产 - 允许代币和兼容ETH的智能合约及钱包服务等第三方个体兼容 以及一些简单的函数功能等等。9 s2 j# t) F7 n/ ^$ ]+ ^- q
- pragma solidity >=0.7.0 <0.9.0;; F0 Q6 f9 h ?
- / z; Q& m+ [; O3 I) W, V9 M0 K. ~% ?
- //投票实验
- contract Ballot{3 f- ~) o: ]+ W$ Y# E4 h7 f
-
- struct Voter{9 S0 n! q: C' S
- uint weight;//投票(单票)权重
- bool voted;//是否投过票,true为投过票,bool类型默认值为false
- address delegate;//想要委托投票的节点地址,address默认值为0x0- j G5 \5 }# l& E. h' G3 F; y
- uint vote;//想要投票的节点的索引值(被投票节点信息用一维数组proposals存储)8 |/ V# e; `/ g4 u
- }
-
- struct Proposal{//被投票节点的相关参数
- bytes32 name;//被投票节点的姓名标识
- uint voteCount;//累积被投票数
- }* M% B+ |9 ~" Z# [
- ! B% B% b, ?2 \# }4 z! y
- address public chairperson;//投票管理员地址
- mapping(address => Voter) public voters;//地址对投票节点的信息的映射% r }+ [% v2 z' \" n; x$ H
- Proposal[] public proposals;//一维数组存储被投票节点信息
-
- //构造方法、构造函数
- //solidity和其他语言不一样,构造函数仅在部署合约时调用一次,后续调用合约不调用其构造函数& j0 a( U% M3 Q. m
- //且一个合约只能有一个构造函数,不能进行构造函数重载+ W" ^9 k, @, z
- constructor(bytes32[] proposalNames) public{
- chairperson = msg.sender;//将第一次调用该合约的节点设置为管理员
- voters[chairperson].weight = 1;//将管理员投票权置为1
- 5 X1 Z# P' W& V" Z4 M
- for(uint i=0; i<proposalNames.length; i++){
- //将所有被投票人姓名初始化进一维数组proposals,并将其对应票数初始化为0票
- //.push(),括号中内容需要强调数据类型,eg:arr.push(uint(6));
- proposals.push(Proposal({
- name:proposalNames[i],
- voteCount:0
- }));7 g v# |' v; j' P' n8 e: d- x
- }; b" X( P2 _9 x( w
- }& r/ Q4 ^& ~% N8 ^" x% u- r7 K" M
- - d: n; k1 R9 M& @/ m; \1 m
- //由管理员授权可投票节点( ?: ^" M: i# V2 h5 O
- function giveRightToVote(address voter) public{ @' k8 J0 e. F5 W# k
- //require中判断条件为false时,输出字符串"xxx...",异常会被抛出,程序执行会被挂起,$ o+ H+ Y: j! ?! \! s. R+ \( ^
- //未消耗的gas会被退回,合约状态会回退到初始状态
- require(' T! e' P6 Q8 y$ K! A3 P8 _; g. `
- msg.sender == chairperson,"Only chairperson can give right to vote."* {/ X0 q+ Y3 T. w" P C8 @# U
- );//执行此function的节点一定为管理员节点+ ~1 a) r7 T* I# \
- require(( e# v" P% p+ l! `" R
- !voters[voter].voted,"The voter already voted."
- );//若voter没投过票
- require(voters[voter].weight == 0);( `( F3 i8 j2 ^( a
- //调用合约的人是管理员、待授权节点还没投过票、带授权节点投票权重为0时,进行授权! W. k( o$ B$ S. s' S% \' [2 o
- voters[voter].weight = 1;//上述三个require()均成立时,授权票数
- }
-
- //投票授权
- function delegate(address to) public{
- Voter storage sender = voters[msg.sender];
- require(!sender.voted, "You already voted.");+ a$ T5 [: m9 o
- require(to != msg.sender,"Self-delegation is disallowed.");
- //sender满足的条件:要有投票权限、没有投过票、被授权节点不是自己
- + I( |* u* k4 Z; i4 Z: A
- //判断代理节点地址是否为空:address(0)或者address(0x0)% ~. |2 a1 m; D+ Z5 V% b$ O0 F2 ] x
- while(voters[to].delegate != address(0)){6 z7 M8 i( T* G& _' O5 P: m6 C
- to = voters[to].delegate;//找到最终的代理节点+ e: s* F! R9 [
- require(to != msg.sender,"Found loop in delegation.");//若代理节点最终是自己则回退到初始状态
- }
- 4 _/ j2 q4 x2 a; E4 u1 p! g
- sender.voted = true;//票权代理出去,状态改为已投票0 z7 H4 j0 R ^1 A! ~5 t6 |
- sender.delegate = to;//票权代理地址
- Voter storage delegate_ = voters[to];//取出代理节点状态
- 2 i: t% ]% r+ c! w
- //若代理节点已投过票,将新代理的票权投出去,反之则将代理节点票权加和$ Z' z, l% }- t& U
- if(delegate_.voted){
- proposals[delegate_.vote].voteCount += sender.weight;
- }else{
- delegate_.weight += sender.weight; c% y" {5 f$ {9 Q; A; D! z
- }" i, F, n- p* N* E0 {0 h5 e
- }
- 1 f' t' c/ m' z' ?$ Q
- function vote(uint proposal) public{! V; \4 v; Z) }& O, W
- Voter storage sender = voters[msg.sender];//通过地址获取对应投票信息 h( d( _% T5 Z/ l5 `
- require(!sender.voted,"Already voted.");//若sender未投过票
- sender.voted = true;//更改投票状态为已投过票
- sender.vote = proposal;//保存已投票节点
- proposals[proposal].voteCount += sender.weight;//票权加和
- }
-
- //返回票数最多的节点在一维数组proposals中的索引
- function winningProposal() public view returns(uint winningProposal_){
- uint winningVoteCount = 0;
2 I$ k( ~3 g0 h
* d p4 s9 t$ D; D8 S( m% s
solidity投票智能合约代码,亲测可用! P* ]) N3 g+ Y( g: K# L