根据例子学习Solidity
大叔的爸爸
发表于 2022-12-26 19:53:53
177
0
0
以下的合约相当复杂,但展示了很多Solidity的功能。它实现了一个投票合约。 当然,电子投票的主要问题是如何将投票权分配给正确的人员以及如何防止被操纵。 我们不会在这里解决所有的问题,但至少我们会展示如何进行委托投票,同时,计票又是 自动和完全透明的 。1 z/ G/ x$ U ~, q
我们的想法是为每个(投票)表决创建一份合约,为每个选项提供简称。 然后作为合约的创造者——即主席,将给予每个独立的地址以投票权。
地址后面的人可以选择自己投票,或者委托给他们信任的人来投票。
在投票时间结束时,winningProposal()将返回获得最多投票的提案。
pragma solidity ^0.4.22;
/// @title 委托投票. d, a5 z8 N( b- ]' u( w
contract Ballot {0 r% `* \ [$ Q
// 这里声明了一个新的复合类型用于稍后的变量
// 它用来表示一个选民* R2 L7 Y5 T( x" z: D& X
struct Voter {$ `- ]6 ?) [0 r- I
uint weight; // 计票的权重: K1 m7 V0 V9 A E' |
bool voted; // 若为真,代表该人已投票
address delegate; // 被委托人7 [( k; z a/ F; l4 L
uint vote; // 投票提案的索引% O: J, m" i: |
}- t/ r( W$ D% J6 t5 |- W4 C
// 提案的类型
struct Proposal {
bytes32 name; // 简称(最长32个字节)
uint voteCount; // 得票数
}/ ]: y7 g( F& H
address public chairperson;1 N+ H1 a2 n8 q# L
// 这声明了一个状态变量,为每个可能的地址存储一个 `Voter`。
mapping(address => Voter) public voters;
// 一个 `Proposal` 结构类型的动态数组! b" s8 F& Q- R7 e( k/ J
Proposal[] public proposals;; T1 }7 @" l/ Y" e, j) L" U0 ?
/// 为 `proposalNames` 中的每个提案,创建一个新的(投票)表决
constructor(bytes32[] proposalNames) public {
chairperson = msg.sender;
voters[chairperson].weight = 1;$ o/ A. M+ E+ Z( ^9 q
//对于提供的每个提案名称,
//创建一个新的 Proposal 对象并把它添加到数组的末尾。- M6 n8 W4 K8 s/ L
for (uint i = 0; i winningVoteCount) {4 F/ |1 V9 f' y, C
winningVoteCount = proposals.voteCount;: Z; \& P I/ A4 @" c7 l5 D
winningProposal_ = p;
}
}7 U T( c* {) o9 x
}
// 调用 winningProposal() 函数以获取提案数组中获胜者的索引,并以此返回获胜者的名称$ d# M' Z. B5 y4 {2 a- ?2 o7 O; T$ Z
function winnerName() public view7 I$ F$ j4 N) D* M
returns (bytes32 winnerName_)2 K" @# C( y; z, l2 j! J
{8 l% F5 A9 J, H! q ~0 k) b
winnerName_ = proposals[winningProposal()].name;) Q' @% e2 U# @6 p
}- @9 k5 g) p8 a) {* K, v+ N
}/ f- f% O" g8 n8 i
可能的优化
当前,为了把投票权分配给所有参与者,需要执行很多交易。你有没有更好的主意?
秘密竞价(盲拍)
在本节中,我们将展示如何轻松地在以太坊上创建一个秘密竞价的合约。 我们将从公开拍卖开始,每个人都可以看到出价,然后将此合约扩展到盲拍合约, 在竞标期结束之前无法看到实际出价。
简单的公开拍卖
以下简单的拍卖合约的总体思路是每个人都可以在投标期内发送他们的出价。 出价已经包含了资金/以太币,来将投标人与他们的投标绑定。 如果最高出价提高了(被其他出价者的出价超过),之前出价最高的出价者可以拿回她的钱。 在投标期结束后,受益人需要手动调用合约来接收他的钱 - 合约不能自己激活接收
pragma solidity ^0.4.22;
contract SimpleAuction {
// 拍卖的参数。
address public beneficiary;
// 时间是unix的绝对时间戳(自1970-01-01以来的秒数)
// 或以秒为单位的时间段。
uint public auctionEnd;5 v. V0 g4 a& s- D
// 拍卖的当前状态
address public highestBidder;! |3 z, a0 s+ Y
uint public highestBid;
//可以取回的之前的出价
mapping(address => uint) pendingReturns;" Y( L* c9 p: t6 |4 D; W" a2 z
// 拍卖结束后设为 true,将禁止所有的变更
bool ended;
// 变更触发的事件
event HighestBidIncreased(address bidder, uint amount);
event AuctionEnded(address winner, uint amount);
// 以下是所谓的 natspec 注释,可以通过三个斜杠来识别。
// 当用户被要求确认交易时将显示。
/// 以受益者地址 `_beneficiary` 的名义,3 J0 \& ^5 b$ k8 i. x! x; E& L
/// 创建一个简单的拍卖,拍卖时间为 `_biddingTime` 秒。- Z& r$ w. Q7 S' g% o/ d+ `1 N( \! O
constructor(, y: w9 }2 ~- i" d' w7 i; n
uint _biddingTime,* P# r5 Q9 P7 B c
address _beneficiary
) public {
beneficiary = _beneficiary;
auctionEnd = now + _biddingTime;$ c8 X( ~5 Y) \9 y+ k! B/ E" F! \
}
/// 对拍卖进行出价,具体的出价随交易一起发送。6 {; c! x% z& Y
/// 如果没有在拍卖中胜出,则返还出价。
function bid() public payable {" p O. x9 }; \& f% {0 m5 Q+ m' ^( q
// 参数不是必要的。因为所有的信息已经包含在了交易中。3 O: h4 Y) A) L8 f
// 对于能接收以太币的函数,关键字 payable 是必须的。
// 如果拍卖已结束,撤销函数的调用。+ O3 f, ]/ S" t O0 r, J( z& v
require() ]% ?! a0 G; |
now highestBid,
"There already is a higher bid."
);! j8 D& S7 V2 D9 H, c% f& s- N* t
if (highestBid != 0) {
// 返还出价时,简单地直接调用 highestBidder.send(highestBid) 函数,; G+ r8 v3 N2 u" N' }
// 是有安全风险的,因为它有可能执行一个非信任合约。
// 更为安全的做法是让接收方自己提取金钱。
pendingReturns[highestBidder] += highestBid;
} w9 J9 w. W% O
highestBidder = msg.sender;, x0 E% o+ T. V3 f- l0 K
highestBid = msg.value;
emit HighestBidIncreased(msg.sender, msg.value);7 ~# M/ ]: c4 e: t7 G
}# f* m1 z, I4 C
/// 取回出价(当该出价已被超越)
function withdraw() public returns (bool) {
uint amount = pendingReturns[msg.sender];
if (amount > 0) {
// 这里很重要,首先要设零值。/ M! o( v; \3 j2 g. i
// 因为,作为接收调用的一部分,9 o' a e( |5 H- H
// 接收者可以在 `send` 返回之前,重新调用该函数。
pendingReturns[msg.sender] = 0;
if (!msg.sender.send(amount)) {# U( I6 O9 Y( d2 U+ o
// 这里不需抛出异常,只需重置未付款+ S& P( H) c U4 G8 [; _
pendingReturns[msg.sender] = amount; J* _4 U% V6 \) J; \' l
return false;+ B- g) d, y p
}
}# h) H' t( b: d2 y
return true;* y$ x7 J* c+ D. K- u+ ~' w
}3 w& R% v1 ~2 {: B' o
/// 结束拍卖,并把最高的出价发送给受益人6 ]$ S. Q0 F1 [( N7 P
function auctionEnd() public {9 X5 h& ^! n4 R% d: e
// 对于可与其他合约交互的函数(意味着它会调用其他函数或发送以太币),
// 一个好的指导方针是将其结构分为三个阶段:0 S" }# b( g0 J/ }1 h
// 1. 检查条件! S( ?: z4 i, q0 ?
// 2. 执行动作 (可能会改变条件)
// 3. 与其他合约交互1 z6 D( X+ N8 x7 w, w9 G
// 如果这些阶段相混合,其他的合约可能会回调当前合约并修改状态,
// 或者导致某些效果(比如支付以太币)多次生效。
// 如果合约内调用的函数包含了与外部合约的交互,
// 则它也会被认为是与外部合约有交互的。, ?5 U7 T4 w# x; Z2 o" K
// 1. 条件+ J: {( c+ a# k( @% L1 Z( `
require(now >= auctionEnd, "Auction not yet ended.");
require(!ended, "auctionEnd has already been called.");
// 2. 生效
ended = true; \' |/ r+ i0 c- d' _9 j9 O
emit AuctionEnded(highestBidder, highestBid);
// 3. 交互
beneficiary.transfer(highestBid);
}
}: c0 u( c K9 [% G; Y5 Y! n# o4 \
秘密竞拍(盲拍)
之前的公开拍卖接下来将被扩展为一个秘密竞拍。 秘密竞拍的好处是在投标结束前不会有时间压力。 在一个透明的计算平台上进行秘密竞拍听起来像是自相矛盾,但密码学可以实现它。9 `8 c9 b! K- v9 f
在 投标期间 ,投标人实际上并没有发送她的出价,而只是发送一个哈希版本的出价。 由于目前几乎不可能找到两个(足够长的)值,其哈希值是相等的,因此投标人可通过该方式提交报价。 在投标结束后,投标人必须公开他们的出价:他们不加密的发送他们的出价,合约检查出价的哈希值是否与投标期间提供的相同。* r7 }8 q2 a, N* Q/ u T) C- \
另一个挑战是如何使拍卖同时做到 绑定和秘密 : 唯一能阻止投标者在她赢得拍卖后不付款的方式是,让她将钱连同出价一起发出。 但由于资金转移在 以太坊Ethereum 中不能被隐藏,因此任何人都可以看到转移的资金。
下面的合约通过接受任何大于最高出价的值来解决这个问题。 当然,因为这只能在披露阶段进行检查,有些出价可能是 无效 的, 并且,这是故意的(与高出价一起,它甚至提供了一个明确的标志来标识无效的出价): 投标人可以通过设置几个或高或低的无效出价来迷惑竞争对手。0 k1 Q% u& ]3 b0 ]5 R% c* s
pragma solidity >0.4.23 Bid[]) public bids;/ g: Q1 t- m$ |1 K! B; ]
address public highestBidder;
uint public highestBid;
// 可以取回的之前的出价
mapping(address => uint) pendingReturns;
event AuctionEnded(address winner, uint highestBid);' N. P2 g) e2 R4 }2 i
/// 使用 modifier 可以更便捷的校验函数的入参。: {# S4 h: n! s% E: ]" F p; A
/// `onlyBefore` 会被用于后面的 `bid` 函数:/ B$ a& \! Z4 i( g( i
/// 新的函数体是由 modifier 本身的函数体,并用原函数体替换 `_;` 语句来组成的。
modifier onlyBefore(uint _time) { require(now _time); _; }
constructor(
uint _biddingTime,
uint _revealTime,2 ~+ q2 a5 ~# o% _* C0 E! K& {/ K
address _beneficiary2 O& U& w, Z0 ~2 {9 s/ P- w
) public {
beneficiary = _beneficiary;: _$ A! r. w' A: T! o; m9 w$ I- R
biddingEnd = now + _biddingTime;' X7 m2 _% `& S4 g r
revealEnd = biddingEnd + _revealTime;: i5 F5 b; F6 d2 W6 [# J5 C& R
}/ a& _# ~) ~$ v: C2 U1 ]
/// 可以通过 `_blindedBid` = keccak256(value, fake, secret)
/// 设置一个秘密竞拍。
/// 只有在出价披露阶段被正确披露,已发送的以太币才会被退还。# a$ m. Y; q/ q5 R# ^
/// 如果与出价一起发送的以太币至少为 “value” 且 “fake” 不为真,则出价有效。
/// 将 “fake” 设置为 true ,然后发送满足订金金额但又不与出价相同的金额是隐藏实际出价的方法。
/// 同一个地址可以放置多个出价。
function bid(bytes32 _blindedBid)
public
payable
onlyBefore(biddingEnd)
{
bids[msg.sender].push(Bid({
blindedBid: _blindedBid,
deposit: msg.value0 `% y& [& g' Q) g
}));+ a2 w7 r9 V/ j# M. a
}
/// 披露你的秘密竞拍出价。- `1 w c' |9 N6 Y Z9 v4 ~. R6 s
/// 对于所有正确披露的无效出价以及除最高出价以外的所有出价,你都将获得退款。
function reveal(+ f/ q" H7 [7 y
uint[] _values,- @4 L7 W( d: D4 q
bool[] _fake,
bytes32[] _secret
)
public2 D) n5 j% g) ^3 _4 |8 S2 |- x# ]
onlyAfter(biddingEnd)6 V. K. Q. F$ M L$ ?# V
onlyBefore(revealEnd), W+ E4 C* w8 N( h1 t( K
{4 q0 W; g0 k) W. b: t$ M% I* j
uint length = bids[msg.sender].length;
require(_values.length == length);2 S2 V, o1 B. P, G
require(_fake.length == length);
require(_secret.length == length);) Q# l1 t. ]; J/ U; [4 o, A+ u; H4 p
uint refund;
for (uint i = 0; i = value) {
if (placeBid(msg.sender, value))' u1 |5 Q9 C0 L, T
refund -= value;
}
// 使发送者不可能再次认领同一笔订金! e2 E- U8 S. y
bid.blindedBid = bytes32(0);: o {& u H8 P3 n3 |4 u$ g
}/ [5 Y% h: ]2 R
msg.sender.transfer(refund);4 v2 t+ u3 f8 i5 g9 C, B
}" Y; `+ X; W6 j) u; g' A
// 这是一个 "internal" 函数, 意味着它只能在本合约(或继承合约)内被调用
function placeBid(address bidder, uint value) internal
returns (bool success)/ B& o% I& {& e
{& C7 M4 B% `+ E
if (value 0) {$ c. Z2 n; }: X8 n
// 这里很重要,首先要设零值。& h: Y c% q; U' R- N. w i
// 因为,作为接收调用的一部分,
// 接收者可以在 `transfer` 返回之前重新调用该函数。(可查看上面关于‘条件 -> 影响 -> 交互’的标注)
pendingReturns[msg.sender] = 0;
msg.sender.transfer(amount);& e7 `6 f6 k- g9 U' U w! O
}# ~! b: W) Y% V
}
/// 结束拍卖,并把最高的出价发送给受益人4 x& l! W3 w. s# ?* X
function auctionEnd()/ j: k9 C1 Z& F1 {9 p
public
onlyAfter(revealEnd)
{4 _+ Y9 @1 O. h( {% S; [+ D9 i: S
require(!ended);, @- V3 ^! w+ V7 p+ a
emit AuctionEnded(highestBidder, highestBid);+ E6 U& }: n9 K0 h( H
ended = true;
beneficiary.transfer(highestBid);
}. G( L0 \( b, V3 i5 ^ u& }
}
安全的远程购买
pragma solidity ^0.4.22;, _6 G4 ` f/ \* j% i" [0 i5 U
contract Purchase {+ K8 V7 c, W& k/ C( O
uint public value;3 o- ?# ?3 Y7 b* H! I
address public seller;
address public buyer;
enum State { Created, Locked, Inactive }% `" \) n6 c4 n' o/ J+ s
State public state;
//确保 `msg.value` 是一个偶数。" A4 X3 ]2 |* ]
//如果它是一个奇数,则它将被截断。" l& a; m( K+ ^/ E2 |5 k, ?. u
//通过乘法检查它不是奇数。
constructor() public payable {1 E4 K% U/ W3 c3 }4 F
seller = msg.sender;
value = msg.value / 2;
require((2 * value) == msg.value, "Value has to be even.");
}" N$ U4 c4 y% S) m$ U1 w- H3 W- X
modifier condition(bool _condition) {. O, q4 r2 t! ?" m i
require(_condition);
_;
}0 Q3 \9 j& Y; G, b7 Y6 x. r1 t" j
modifier onlyBuyer() {
require(
msg.sender == buyer,
"Only buyer can call this."
);
_;
}
modifier onlySeller() {
require(
msg.sender == seller,
"Only seller can call this."( q- T: h0 u. W" }7 s7 i: E$ A
);
_;: u0 n) ^1 ]* G
}1 T# F* f- n7 R2 [
modifier inState(State _state) {
require(% J. g4 q: S3 S. U. X2 C8 R
state == _state,( A1 a- ]+ a& @% c/ U- b! {8 I- C
"Invalid state."; T* I+ I" m8 P% B# s
);
_;/ b- |: ]9 S7 A, v
}* V5 ?$ Y; t4 e% n' C
event Aborted();
event PurchaseConfirmed();
event ItemReceived();; {' P+ I, k- h& E
///中止购买并回收以太币。
///只能在合约被锁定之前由卖家调用。* p# J6 y: C6 ]4 \* ]% K
function abort()) Y( B6 E/ f5 h, N8 |4 d, h1 X) W
public
onlySeller& `, M$ `7 |' X6 b1 x
inState(State.Created)
{
emit Aborted();
state = State.Inactive;1 \; p: ~0 A% v1 M+ P
seller.transfer(address(this).balance);& Y' m9 b1 S; U
}
/// 买家确认购买。/ {2 F7 Q7 A5 |7 k6 U% b) q
/// 交易必须包含 `2 * value` 个以太币。, W& i; F' v3 y. f, Y) }
/// 以太币会被锁定,直到 confirmReceived 被调用。& C( O& |: \7 ]# \
function confirmPurchase()
public* c# y s) U) h* M
inState(State.Created)
condition(msg.value == (2 * value))9 [8 v; L" M6 Z* ? C; B
payable Y( E0 o6 C/ ]
{
emit PurchaseConfirmed();: v9 K: {# X- ~$ W% d* i
buyer = msg.sender;
state = State.Locked;# a; j0 K$ y% X+ S" l; c* l
} C" Y( N) f2 ~ L
/// 确认你(买家)已经收到商品。/ _" D9 y% c6 ~) l3 ~% L3 d* i) P/ w
/// 这会释放被锁定的以太币。" e+ x1 P7 T) m8 {
function confirmReceived()
public
onlyBuyer
inState(State.Locked)
{; R5 t I( x) z5 T0 n
emit ItemReceived();( ]. Z) N- Q1 p+ J( r! t1 A
// 首先修改状态很重要,否则的话,由 `transfer` 所调用的合约可以回调进这里(再次接收以太币)。
state = State.Inactive;, Z( q% t7 G0 b1 r! \- r
// 注意: 这实际上允许买方和卖方阻止退款 - 应该使用取回模式。
buyer.transfer(value);2 X9 V: p! G3 @/ t/ i
seller.transfer(address(this).balance);, T' z* z6 d/ J5 ^
}
}
成为第一个吐槽的人