智能合约开发环境搭建及Hello World合约
945坏男人
发表于 2022-11-7 00:10:06
173
0
0
就先学习任何编程语言一样,入门的第一个程序都是HelloWorld。今天我们来一步一步从搭建以太坊智能合约开发环境开始,讲解智能合约的HelloWorld如何编写。
* E' c9 @6 p3 R/ Z
开发环境搭建- _. U2 ~8 X3 H* ]: N. s/ [
Solidity安装" R. Z& a1 n6 E; R# u
强烈建议新手使用Remix-SolidityIDE来进行开发。
! M; t1 f8 j0 Z6 c7 _% A
Remix是一个基于浏览器的Solidity,就可以不用安装Solidity,本文的HelloWorld教程也将基于RemixSolidityIDE来进行。
+ }# Q2 q) L5 r; n- I3 j# R. ^: U# z
如果你想自己安装请参考Solidity安装指引。
8 i ~$ c" u' ?0 _
更新,开发环境搭建还可以看另一篇文章:搭建智能合约开发环境RemixIDE及使用。8 H4 A4 ^, q/ A7 ]; r6 o
geth安装6 q. V6 |* k6 ^" _+ m
- g1 ^3 N* ~0 p& O
Mac下安装命令如下:其他平台参考:geth官方安装指引
brewtapethereum/ethereum
; h1 J4 c; J; F1 l
brewinstallethereum) W5 X" Q. Y1 b$ s" L
brew是Mac下的包管理工具,和Ubuntu里的apt-get类似
安装完以后,就是把geth控制台启动。
启动环境
* _( |2 f# _. ~8 ]; e( N
在入门篇讲过,geth是一个以太坊客户端,现在利用geth启动一个以太坊(开发者)网络节点。! h+ r6 @, ?# M$ {5 x
geth--datadirtestNet--devconsole2>>test.log
执行命名后,会进入geth控制台,这时光标停在一个向右的箭头处,像这样
命令参数说明6 N* _) f; n- X& b( M
–dev启用开发者网络(模式),开发者网络会使用POA共识,默认预分配一个开发者账户并且会自动开启挖矿。* k; K. w! w$ k8 [9 z; p2 S
9 N- m# B) D& m8 k" N
–datadir后面的参数是区块数据及秘钥存放目录。
7 e# v4 l- o4 ]: z6 o% i( b8 B
第一次输入命令后,它会放在当前目录下新建一个testNet目录来存放数据。
console进入控制台
2>>test.log表示把控制台日志输出到test.log文件
为了更好的理解,建议新开一个命令行终端,实时显示日志:
tail-ftest.log
& z" s2 I; a) k) U! D( n
准备账户- d% {) z( D3 K: m$ j
- j* u7 r4 {: x! N+ v
部署智能合约需要一个外部账户,我们先来看看分配的开发者账户,在控制台使用以下命令查看账户:. P0 X) v( ?6 ~, V; T: `; X
>eth.accounts# o+ r+ S2 R8 i- g4 y+ S$ b
" \) }% P7 k' w8 ]5 z
回车后,返回一个账户数组,里面有一个默认账户,如:
1 q* v2 p5 f1 I% K' P. s. C
也可以使用personal.listAccounts查看账户,: r v( Z0 I- Q4 e7 `
再来看一下账户里的余额,使用一下命令:
* j3 ~1 R4 z# @
>eth.getBalance(eth.accounts[0])6 v9 r: Z( X( q% g- q# b; G
& e' p1 \5 }7 Q3 u* g
**eth.accounts[0]**表示账户列表第一个账户
回车后,可以看到大量的余额,如:
( N1 |! m) ?0 x4 d
1.15792089237316195423570985008687907853269…e+77
& ^' s7 Z4 c# P- N4 Y- _0 ?
开发者账户因余额太多,如果用这个账户来部署合约时会无法看到余额变化,为了更好的体验完整的过程,这里选择创建一个新的账户。
创建账户
! ]" W4 M: q) Z- ]) u
使用以下命令创建账户:
>personal.newAccount("TinyXiong")
TinyXiong为新账户的密码,回车后,返回一个新账户。& u! ~$ ?: x4 C8 Y$ `# x. p
/ `6 c: ~6 f& p5 j; m
这时我们查看账户列表:0 J7 I+ ]; g( \* L9 X! y. X3 d* V
>eth.accounts a3 _9 J9 e% l) }/ J" B
可以看到账户数组你包含两个账户,新账户在第二个(索引为1)位置。8 Z3 V! N: q8 D' k+ m' B
现在看看账户的余额:
>eth.getBalance(eth.accounts[1])
1 J$ f& d. C! R3 t: W9 I
0! [, `4 s; q! A* u) P
. W3 @: K* P/ N7 y
回车后,返回的是0,新账户是0。结果如:( |& }. j% O1 V9 z$ p! v/ r8 Q* P
给新账户转账) g9 F: W" u! N8 n! w
我们知道没有余额的账户是没法部署合约的,那我们就从默认账户转1以太币给新账户,使用以下命令(请使用你自己eth.accounts对应输出的账户):
9 k, v4 `- S/ n/ G% \6 P4 B
eth.sendTransaction({from:'0xb0ebe17ef0e96b5c525709c0a1ede347c66bd391',to:'0xf280facfd60d61f6fd3f88c9dee4fb90d0e11dfc',value:web3.toWei(1,"ether")})
. d) @2 D, n( C* c5 a3 F
在打开的tail-ftest.log日志终端里,可以同时看到挖矿记录% K0 ?+ \3 ^3 X$ |7 H
1 q" w% x0 ]6 x( |/ H
再次查看新账户余额,可以新账户有1个以太币
' L% g4 x S8 K. k4 y( }
解锁账户
在部署合约前需要先解锁账户(就像银行转账要输入密码一样),使用以下命令:* ]8 f8 A- t" g! E* {7 o. H
$ B4 G z; L% o* i
personal.unlockAccount(eth.accounts[1],"TinyXiong");
“TinyXiong”是之前创建账户时的密码0 o- _+ [6 Y2 T7 @; g! Z+ j1 y
, T+ B* Z1 q! u
解锁成功后,账户就准备完毕啦,接下来就是编写合约代码。
& ^; j; j0 G4 u3 v# ]1 f7 j! C$ I
编写合约代码: a) w, S* v( @' ]0 P: g9 K. X) b
+ a e& q, S* {. a- B
现在我们来开始编写第一个智能合约代码,solidity代码如下:
; \1 m. z9 x) }- V8 V) `& g
pragmasolidity^0.4.18;
contracthello{
e; Z' }8 D/ y, P8 \) q
stringgreeting;8 W# B- b6 N# D+ @
8 q: {4 D/ R, X$ O9 M$ H
functionhello(string_greeting)public{
; r5 H3 a! \: V
greeting=_greeting;9 c/ H! B h( v ]% T6 d/ p4 n7 A
}5 \( ~6 C @: {
' j' \; H5 J, S j; D1 E
functionsay()constantpublicreturns(string){
returngreeting;
}
}- w4 I; c. H1 d; D2 T6 K! P
简单解释下,我们定义了一个名为hello的合约,在合约初始化时保存了一个字符串(我们会传入helloworld),每次调用say返回字符串。
& O0 l$ G2 S; i) k9 N4 b" U
把这段代码写(拷贝)到Browser-Solidity,如果没有错误,点击Details获取部署代码,
如
在弹出的对话框中找到WEB3DEPLOY部分,点拷贝,粘贴到编辑器后,修改初始化字符串为helloworld。
- q6 [9 Y0 Q+ ^2 |2 b
solidity在博文写作时(2017/11/24),版本为0.4.18,solidity发展非常快,solidity版本之间有可能不能兼容,这是你可以在Browser-Solidity的Settings里选择对应的编译器版本。* X: N8 k- O' A+ n- Y% @
+ {) t- C O5 Q
Browser-Solidity也不停的更新中,截图可能和你看到的界面不一样。
部署合约
- f& [/ @: q. f. K" m O& W1 J
Browser-Solidity生成的代码,拷贝到编辑器里修改后的代码如下
% R) M) L. m: s% V# K8 m3 P0 O
var_greeting="HelloWorld";$ l8 t5 P% l7 N# P" q/ F) O7 G
varhelloContract=web3.eth.contract([{"constant":true,"inputs":[],"name":"say","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"inputs":[{"name":"_greeting","type":"string"}],"payable":false,"stateMutability":"nonpayable","type":"constructor"}]);9 {- N/ [$ m2 r, M1 L& A
varhello=helloContract.new(9 Q) d: _5 U4 Z+ Y
1 N# q# Y# t/ |
_greeting,: U4 E# |/ n4 J( y, \7 R9 A2 G
8 [" k2 q4 j& k& S. x8 s
{
from:web3.eth.accounts[1],
" E8 p' @! L4 z% e3 m& H+ y9 R9 H2 Z
data:'0x6060604052341561000f57600080fd5b6040516102b83803806102b8833981016040528080518201919050508060009080519060200190610041929190610048565b50506100ed565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061008957805160ff19168380011785556100b7565b828001600101855582156100b7579182015b828111156100b657825182559160200191906001019061009b565b5b5090506100c491906100c8565b5090565b6100ea91905b808211156100e65760008160009055506001016100ce565b5090565b90565b6101bc806100fc6000396000f300606060405260043610610041576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063954ab4b214610046575b600080fd5b341561005157600080fd5b6100596100d4565b6040518080602001828103825283818151815260200191508051906020019080838360005b8381101561009957808201518184015260208101905061007e565b50505050905090810190601f1680156100c65780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6100dc61017c565b60008054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156101725780601f1061014757610100808354040283529160200191610172565b820191906000526020600020905b81548152906001019060200180831161015557829003601f168201915b5050505050905090565b6020604051908101604052806000815250905600a165627a7a723058204a5577bb3ad30e02f7a3bdd90eedcc682700d67fc8ed6604d38bb739c0655df90029',
gas:'4700000'3 @2 g# H4 v2 V2 |! S* D7 Z7 |
},function(e,contract){- G- k1 X8 _( b5 M" x# m
' |! S* T2 S0 i' V/ L2 H7 t
console.log(e,contract);0 b y; ]( a. }& l4 g/ K
9 y7 v& U, {9 T; ~8 a V0 k1 y; y3 Y
if(typeofcontract.address!=='undefined'){
, C F, j- F N2 Z- x; g8 k* q
console.log('Contractmined!address:'+contract.address+'transactionHash:'+contract.transactionHash);; H! b& G8 C% J( n& a k
' x# ~3 J2 a7 H& H3 ^6 i6 V# P
}
});
, L, N, e) g8 `! y2 X; X
第1行:修改字符串为HelloWorld
第2行:修改合约变量名3 T6 e) f6 ?& S/ I; K* O0 j
第3行:修改合约实例变量名,之后可以直接用实例调用函数。8 a& g E' f7 G3 S1 S( N
第6行:修改部署账户为新账户索引,即使用新账户来部署合约。( l6 U) p- o+ D; h8 Y9 \- y
& z3 X2 S6 d$ l: x7 Q
第8行:准备付的gas费用,IDE已经帮我们预估好了。
第9行:设置部署回调函数。
拷贝回geth控制台里,回车后,看到输出如:
Contractmined!address:0x79544078dcd9d560ec3f6eff0af42a9fc84c7d19transactionHash:0xe2caab22102e93434888a0b8013a7ae7e804b132e4a8bfd2318356f6cf0480b3- Z/ O) Z% L# ~3 m
说明合约已经部署成功。" x5 g3 c+ T, a9 G* w
在打开的tail-ftest.log日志终端里,可以同时看到挖矿记录
3 k# K e, P* g, b8 F
现在我们查看下新账户的余额:; H1 }; r. [7 [! l2 [! E5 ^9 M
>eth.getBalance(eth.accounts[1])
是不是比之前转账的余额少呀!
0 y: b9 W+ N3 B
运行合约
>hello.say()5 @7 S4 r+ N7 W4 n6 q- H
2 ?" E+ C1 @* Y# e. k0 A
"HelloWorld", g8 k& E& a/ e4 E) y- l4 ?
输出HelloWorld,我们第一个合约HelloWorld,成功运行了。
运行截图如下:
' g2 V% ?3 a! g, l& C
第一个合约的意义更重要的是体验智能合约开发流程,对于初学者一些可以选择先放弃一些细节,开发流程打通之后,可以增强信心进行下一步的学习。# @0 C$ @# x" t* V0 _2 N* L5 h
2 S# E- y; V p5 r/ ]0 K
作者:@xilibi2003
成为第一个吐槽的人