以太坊开发演练从智能合约开始
人民干脆面
发表于 2022-12-2 03:37:35
103
0
0
/ E1 i2 U- n, v8 |
一些人在测试网络中部署智能合约,另一些人告诉你要读黄皮书,而另一些人则建议你使用truffle套件,因为它很有帮助。但其实你并不知道该做什么,也不知道这些东西放在一起是怎么用的。
# x" T( }1 d& P) S% M6 w
如果这是你读到的第一篇关于以太坊或区块链生态系统的文章,你会爱上以太坊的!虽然“专家们”在twitter上互掐,有很多不安全的标准和协议,未经测试有bug的开发工具……并不是所有的问题都已经解决,但每个人都在朝着不同的方向前进探索,任重而道远。机构、银行和政府的未来将由这些疯狂的开发者决定!这真是太棒了。
不管怎样,别担心,我会努力把本系列教程中所有的信息点串连起来,带你走进智能合约和dapps的开发世界,向你展示各部分是如何配合工作的。3 m" e) G$ J( b' m# f) C8 u
虽然我不会对每一个细节进行阐释,但我会给你一些阅读材料的链接,这些材料能帮助你更好地理解某些概念,这要靠你自己研究并了解所有的细节。这个系列的目的就是以最简单的方式,帮助你更好地理解各部分是如何协同工作的,就像一个朋友向你解释说明那样。
l5 O, c$ }+ a2 k
我甚至不知道以太坊是什么
: u5 s4 W% R% l3 \
以太坊的官方网站告诉我们:) S% \ C8 q. z! @% R
) }- i; y: {0 Q. h5 } H0 v2 y
以太坊是一个运行着智能合约的分布式平台:应用程序完全按照程序运行,不存在故障、审查、欺诈或第三方干预的可能性。
换句话说,以太坊给予了我们使用区块链技术来验证我们运行的代码的执行情况的能力。1 O2 K: L! S) Y6 M/ K. r4 q
# n; n1 r3 b2 T( N
如果你不知道区块链、以太坊、比特币、加密货币这些词的含义,我建议你听听TimFerriss采访NickSzabo和NavalRavikant的精彩播客:沉默的加密货币大师——NickSzabo。
智能合约是什么?
在以太坊,智能合约是可以处理资金的脚本。就是这么简单。
这些合约由我们称为“矿工”的参与方强制执行和证明。矿工其实是多台电脑,它们把一项交易(执行智能合约,支付加密货币等)添加到一个公开分类账上,我们称之为区块。多个区块构成一个区块链。8 J9 k/ N+ N6 Z- o( V- D
5 D" [7 V8 g) V- b
我们给这些矿工支付一种叫作“Gas”的东西,它是运行一份合约的成本。当你发布一份智能合约,或者执行一份智能合约,或者把钱转到另一个账户,你就要支付一些被转换成Gas的以太币。
在深入智能合约的开发之前5 r$ H. A9 P% N2 {" }+ g; V" q
在本教程中,我会假设你已经在软件开发方面有了一定的基础,并且具备了Javascript和Node.JS的基本知识。否则,你很快会看不懂。Solidity是我们用来开发的编程语言,它与Javascript的语法非常接近,而且由于以太坊周围很多的开发工具都是基于Javascrip和Node.JS.的,所以如果你对两者都已经很熟悉的话,那你就更容易理解。这不是针对初学者的教程,如果你不理解一些概念,我会为给你一些替代教程的链接,剩下的部分靠你自己研究吃透。
% R* A! p5 F. `) e _% ?! t
智能合约开发
, Q; N3 I2 }: L% ^" ?/ D7 X
现在,我们开始有趣的部分,正如我所说的,Solidity与Javascript很接近,但它们并不相同。关于Solidity,我很遗憾地告诉你,作为前端开发人员,你不能在一段代码上强加JQuery并希望它能起作用,你不能忽略安全性,不能“另辟蹊径”。我们会在讨论安全性、开发模式和防止代码故障的时候知晓原因。
现在,关于我们的第一个例子,我正在考虑一个由电影《时间规划局》启发的脚本。电影中,人们生活在一个反乌托邦式的未来,改用时间作为货币流通。他们可以通过掰手腕的方式赢取对手的时间(他们的“手臂”上存储着时间,输方的时间将会传送给赢家),我们也可以这么做!用智能合约以角力(Wrestling)的方式赚钱。5 _, T9 T7 m5 G d5 U' o0 k8 H
* U4 U) }7 Y7 `5 y7 x9 g, e/ o
别担心,我会在Github上为你提供接下来我们会看到的所有脚本。0 `- G- d: z5 X0 p/ D
/ r/ Y+ b2 F% W/ l
另外,如果这是你第一次处理智能合约,如果你现在不了解所有的事情那也没关系,你需要练习、阅读文档和做一些研究来熟悉Solidity。1 H0 i# V+ l3 U" M# H
代码
' L) M6 ?4 m$ k6 x# e# G
现在话题转到写代码上,首先,solidity脚本的基础是下面这段代码,pragma指令会告诉编译器我们正在使用的Solidity版本,以及我们的合约名称,是一种与Javascrip上的类(class)相似的结构。这就是“Wrestling(角力)”。
pragmasolidity^0.4.18;+ x7 t1 I& h k3 \# N. F' v
contractWrestling{
//ourcodewillgohere
}4 Q7 a& v; ]/ J8 P% N
: c3 A/ F: _* W; m. L D, R4 I
我们需要两个参与者,所以我们要添加两个保存他们账户地址的变量(他们钱包的公钥)。' m' ]6 Z( F9 K. [2 s& T
addresspublicwrestler1;) Q ]% j, X. R$ C
addresspublicwrestler2;& s: s" Z9 L# H3 g* ^$ }* z
, ^5 ?2 w2 u. S h" W0 O- ?$ Z
在我们的小游戏中,每一轮的比赛,参与者都可以投入一笔钱,如果一个人投入的钱是另一个人的两倍(总计),那他就赢了。& b/ g- R# D9 U6 c9 I6 H/ e
boolpublicwrestler1Played;
boolpublicwrestler2Played;% q: |6 R5 X* m0 Q7 v
8 h0 ]. X' _7 v. Z- F
uintprivatewrestler1Deposit;
2 o9 q, ]$ B& `6 U( V$ i, y" c
uintprivatewrestler2Deposit;8 m R9 P2 L! s s
boolpublicgameFinished;
addresspublictheWinner;
uintgains;; e: C# |4 J6 H3 T2 e! Q3 k
关于公钥/私钥的一件重要的事项是,即使一个变量是私有的,但并不意味着其他人不能读取它的内容,它只是意味着它只能在合约中被访问,但实际上,由于整个区块链存储在许多计算机上,所以存储在变量中的信息总是可以被其他人看到,这是你应该记住的第一个安全考虑。
另一方面,编译器会自动为公共变量创建getter函数。为了使其他的合约和用户能够更改公共变量的值,你需要创建一个setter函数。
6 P, x9 ?, F9 s' g$ S. N
现在我们将为游戏的每一步添加三个事件。开始,参与者注册;游戏期间,登记每一轮赛果;最后,其中一位参与者获胜。事件是简单的日志,可以在分布式应用程序(也称为dapps)的用户界面中调用JavaScript回调函数。在开发过程中,事件甚至可以用于调试的目的,因为它不同于JavaScript在Solidity上的console.log()函数。
eventWrestlingStartsEvent(addresswrestler1,addresswrestler2);3 {0 g0 t& m. V+ U
eventEndOfRoundEvent(uintwrestler1Deposit,uintwrestler2Deposit);6 T$ L/ I% S: f3 n
eventEndOfWrestlingEvent(addresswinner,uintgains);& I/ C7 w* C* f8 l) w! f, N B8 U
0 H( e h& A- }6 [
现在我们将添加构造函数,在Solidity中,它与我们的合约具有相同的名称,并且在创建合约时只调用一次。
在这里,第一位参与者将是创造合约的人。“msg.sender”是调用该函数的人的地址。, x+ J8 v& ~1 W8 {" `
) Y; z5 M; Z1 j
functionWrestling()public{! Z% a: u. ]9 ]& U. J" F
wrestler1=msg.sender;
}7 M: K& O$ R0 b$ q1 M2 x6 D/ o9 e3 Q
接下来,我们让另一个参与者使用以下函数进行注册:
functionregisterAsAnOpponent()public{
2 |1 o7 y' y, `7 v3 m) g
require(wrestler2==address(0));, Y; ] `1 t. @, \5 @: d
wrestler2=msg.sender;
WrestlingStartsEvent(wrestler1,wrestler2);
}- W7 d' l5 q4 D) Z9 y v' M! s
" e: Q! d6 L0 v6 D" I
require函数是Solidity中一个特殊的错误处理函数,如果条件不满足,它会回滚更改。在我们的示例中,如果变量参与者2等于0x0地址(地址等于0),我们可以继续;如果参与者2的地址与0x0地址不同,这就意味着某个玩家已经注册为对手,所以我们会拒绝新的注册。% v& y) N! m3 }3 \) ] K# ~- C v
再次强调,“msg.sender”是调用该函数的帐户地址,并且当我们触发一个事件,就标志着角力的开始。8 y0 L3 V( R2 Q8 w1 x
现在,每一个参与者都会调用一个函数,“wrestle()”,并投入资金。如果双方已经玩过这场游戏,我们就能知道其中一方是否获胜(我们的规则是其中一方投入的资金必须是另一方的双倍)。关键字“payable”意味着函数可以接收资金,如果它不是集合,函数则不会接受以太币。“msg.value”是发送到合约中的以太币的数量。& k D) Y: }7 o; Q- t+ Q
functionwrestle()publicpayable{2 e+ z* w5 s" `# Z$ W
require(!gameFinished&&(msg.sender==wrestler1||msg.sender==wrestler2));. ] A% d y$ U
! S- [* N* N( ?5 P K5 Y
if(msg.sender==wrestler1){1 f( H2 r5 C+ A+ R' o2 k* B* q
require(wrestler1Played==false);
3 {5 e9 u7 o% W: g1 i/ e" C) d
wrestler1Played=true;
( S( b& j3 j6 U$ w+ k
wrestler1Deposit=wrestler1Deposit+msg.value;& Q; I% P3 i! Z3 n+ X! C
" r0 d3 |' L/ H0 [) Q6 {6 C
}else{8 e0 k" ~) t- E$ V
: t- \) v Y3 m% h6 s
require(wrestler2Played==false);# c9 T0 n" ~2 t: d3 h6 b. d6 Z, l
wrestler2Played=true;
wrestler2Deposit=wrestler2Deposit+msg.value;7 q) x$ p- T& {2 [
) }8 r5 B) d+ E8 L1 |% r
}
8 C0 l( Q* S7 l+ p' _( Y
if(wrestler1Played&&wrestler2Played){
- j; Y7 p2 c( T' j/ D- X5 F
if(wrestler1Deposit>=wrestler2Deposit*2){
0 W4 t+ @' W0 X( F2 n, [ B$ ]9 c
endOfGame(wrestler1);+ R( r C" _+ [( a' M
* F3 N% u# f4 a3 T( C
}elseif(wrestler2Deposit>=wrestler1Deposit*2){5 M* J/ n2 W* r/ G& a: |6 c" `
5 b7 N* |& \! K3 v- G
endOfGame(wrestler2);+ p; n: g% R8 _2 S9 f) N: }
}else{
+ T9 N1 P' O% g$ f# L
endOfRound();9 r5 @ M( u/ ~( H0 o7 }* X
) {- F/ y* Z) u4 e7 I6 i) X4 @/ B
}
}
}; g$ Q0 R, E( j0 o8 T3 |
然后,我们添加endOfGame()和endOfRound()函数。关键字“internal”与“private”是一样的,唯一的区别是internal函数可以由其他合约继承(因为Solidity与其他面向对象语言相似),而private函数不能继承。6 [, l9 @8 E0 u* W2 f
functionendOfRound()internal{
wrestler1Played=false;9 ~2 I) l" }" V3 i! u9 I5 i; r9 f
wrestler2Played=false;, {2 [0 ^+ b. D4 y
EndOfRoundEvent(wrestler1Deposit,wrestler2Deposit);6 D: o: O* F1 m' q/ k1 E2 _ w
9 V3 q$ C& h; m3 [
}0 I# W% e8 V& H7 ?, q& ^7 o
; t7 h8 G; C* U) V( U8 ?5 ?
functionendOfGame(addresswinner)internal{
gameFinished=true;
" v$ S, f- @. D# w* s( J m3 J
theWinner=winner;3 q/ v8 l2 Z) [) Y9 l: k; o
gains=wrestler1Deposit+wrestler2Deposit;( _1 P$ H& n8 c9 W5 S3 C! B; H
" S/ S$ ^8 L1 @- w
EndOfWrestlingEvent(winner,gains);
} ?, H& I( i: k6 C0 P0 Y% o4 F
请注意,我们不是直接把钱交给赢家,在此情况下这并不重要,因为赢家会把该合约所有的钱提取出来;而在其他情况下,当多个用户要把合约中的以太币提取出来,使用withdraw模式会更安全,可以避免重入。: m) D: h2 W/ ?) a: t
1 M, U0 ]) ^3 `) x' x
简单地说,如果多个用户都可以从合约中提取资金,那么任谁都能一次性多次调用withdraw函数并多次得到报酬。所以我们需要以这样一种方式来编写我们的取款功能:在他继续得到报酬之前,他应得的数额会作废。% ^% l& ~5 z& l6 s8 L; ]7 |
- `( ?5 G+ F6 H8 b* x- ?$ V, Z
它看起来像这样:
functionwithdraw()public{
require(gameFinished&&theWinner==msg.sender);
. L' H# e$ f6 t; w8 U7 D9 y% Z
uintamount=gains;1 I- _% R' }& D; v' A6 N; L
gains=0;
msg.sender.transfer(amount);
}
8 L( O3 u' D" M+ A7 p% z9 o
就是这样。你可以在以下链接中找到整个代码段:
7 a4 r, ~) f6 S- y
devzl/ethereum-walkthrough-1- O4 {6 h# \3 s0 f8 M' v
% S5 f P2 R7 D( Z( ~
使用IDE4 m# L7 {6 G0 V" H; Y
9 k _- }8 k. S' o' g: U O
现在复制代码段,在浏览器的新标签页中打开RemixIDE:% P8 q" e# i: c
你可以直接在浏览器上使用Remix,它有很多有意思的功能,这些功能你在别的地方几乎找不到。
1 V( {7 m7 A4 N" ^. @/ c
点击页面左上方的“+”按钮,创建一个名为“Wrestling.sol”的新文件,然后粘贴你在上面的github中找到的代码。
4 i2 X1 R8 m: z( w
最后会有一些语法高亮显示。Github暂不支持Solidity.sol扩展。
, N) B4 T: p8 Y2 V% l4 v, D j2 T
在页面的右侧,你可以找到很多有意思的选项卡,比如Analysis,它会显示错误和建议。这个工具留给你好好探索。我还是想向你演示一下,即使在接下来的教程中我们会使用其他的工具。但正如我之前所说的,本系列教程的目标是连接这些信息点并向你展示它们是如何在一起协同工作的,你可以自由决定是否在浏览器上使用IDE。如果你愿意的话,邀请你读一下Remix的文档。
或者,你可以阅读这些文档来熟悉Solidity。
在接下来的教程中,我们将了解如何将智能合约部署到两种测试网络中,探索truffle,ganache和geth以及它们是如何一起协同工作的。
成为第一个吐槽的人