智能合约开发环境搭建及Hello World合约
945坏男人
发表于 2022-11-7 00:10:06
120
0
0
* e7 }) n2 k; K6 t+ G' ]8 G
就先学习任何编程语言一样,入门的第一个程序都是HelloWorld。今天我们来一步一步从搭建以太坊智能合约开发环境开始,讲解智能合约的HelloWorld如何编写。5 r: T6 b% L7 Y( K
0 T9 k4 F7 l6 U1 d
开发环境搭建4 E! f$ o3 p, b3 U& B
1 I% y1 E. X# U \) [' _9 n
Solidity安装
强烈建议新手使用Remix-SolidityIDE来进行开发。! \3 [ J' W+ a1 U' u% W0 |
Remix是一个基于浏览器的Solidity,就可以不用安装Solidity,本文的HelloWorld教程也将基于RemixSolidityIDE来进行。; J0 s& u7 [9 C; X
如果你想自己安装请参考Solidity安装指引。
% H8 R' t2 q7 C
更新,开发环境搭建还可以看另一篇文章:搭建智能合约开发环境RemixIDE及使用。
geth安装 y( V& y$ m& F2 i
# l; @: |4 C; u E
Mac下安装命令如下:其他平台参考:geth官方安装指引+ b$ }( L5 V" g
brewtapethereum/ethereum" n' w# H1 z; B& g: a/ c8 y* y4 r
* `7 E& |+ L+ u" N) ^
brewinstallethereum
brew是Mac下的包管理工具,和Ubuntu里的apt-get类似0 x) m0 {+ L. K) O# }+ G9 r3 `
. S3 \! I6 N. b/ i# {4 F1 m
安装完以后,就是把geth控制台启动。3 r, n' ]0 S& [3 @/ e
$ w) V8 ` ]8 o3 D2 \( e
启动环境1 u9 U& f0 g- \% P
; K8 I. X Z& A/ x9 v
在入门篇讲过,geth是一个以太坊客户端,现在利用geth启动一个以太坊(开发者)网络节点。* l' t' f/ n: s7 L$ i5 T# j) D
geth--datadirtestNet--devconsole2>>test.log
执行命名后,会进入geth控制台,这时光标停在一个向右的箭头处,像这样
6 }( X2 T! G& r% r8 q( M
命令参数说明( v) b# i$ ]( U
, j+ x8 M: J8 R4 r* ?2 c- t
–dev启用开发者网络(模式),开发者网络会使用POA共识,默认预分配一个开发者账户并且会自动开启挖矿。
$ p) u B. G- E2 t% n8 U* n
–datadir后面的参数是区块数据及秘钥存放目录。5 z5 n( K0 D: C& L9 g
第一次输入命令后,它会放在当前目录下新建一个testNet目录来存放数据。+ i7 Y4 i' y. i- C
0 f- O9 F* B9 K: Q5 b4 ^- v
console进入控制台; U0 a" B! X0 T1 n+ \+ i4 T
5 I' B+ f/ L5 J' l3 k
2>>test.log表示把控制台日志输出到test.log文件5 r. Q5 s0 {' K3 V0 Y, S4 O
" W6 ]% T$ I! Z* C7 o5 y
为了更好的理解,建议新开一个命令行终端,实时显示日志:% Z" {2 c! t+ J$ F% Q0 i
tail-ftest.log3 l: v( ^( C" ]! P6 Z4 O5 P I& C
/ ^% a; q- f$ R6 c- v( r: K5 n
准备账户
部署智能合约需要一个外部账户,我们先来看看分配的开发者账户,在控制台使用以下命令查看账户:
>eth.accounts
回车后,返回一个账户数组,里面有一个默认账户,如:
3 K/ y# c% a& p' @3 e3 I
也可以使用personal.listAccounts查看账户,
再来看一下账户里的余额,使用一下命令:' f, U1 g8 x0 ], Z, }! v0 \
m& F) X. f8 L
>eth.getBalance(eth.accounts[0])# a4 m2 j5 Z; V* A1 G2 r
3 |; o4 J9 G3 D U3 U
**eth.accounts[0]**表示账户列表第一个账户
回车后,可以看到大量的余额,如:2 N* C: Q: K9 Y5 y; ~9 p8 O9 ]
" F8 C% u U0 h* P
1.15792089237316195423570985008687907853269…e+772 \3 t8 O6 O R' ]4 _0 l n
开发者账户因余额太多,如果用这个账户来部署合约时会无法看到余额变化,为了更好的体验完整的过程,这里选择创建一个新的账户。' M c% t }' f# ~
8 i6 `5 }' F" A2 G$ U+ q4 ~& `# X$ z
创建账户0 X8 R# \: X( O. _: f
使用以下命令创建账户:$ ^- w% n9 [' x& [4 y$ r
/ B- w; R4 C3 a
>personal.newAccount("TinyXiong")2 T: H8 S. n0 y
TinyXiong为新账户的密码,回车后,返回一个新账户。
这时我们查看账户列表:
>eth.accounts/ W: Z: Y' d8 _) ]$ K) t) ?; n% b
/ N' K* g# @' J9 R6 w4 ~ \. a( Z! s
可以看到账户数组你包含两个账户,新账户在第二个(索引为1)位置。
, ?, G9 a5 D/ Q+ d5 w
现在看看账户的余额:' o. F% w. [1 f+ a" e- S# j# [
$ p7 e1 a+ I# ~
>eth.getBalance(eth.accounts[1]): U2 v6 [% J" r# {# [0 `8 c% l
& `$ w* ?, A: J
04 W/ F. X- p% _( C7 i7 e
$ s2 h* T& d: S$ s( p
回车后,返回的是0,新账户是0。结果如:
6 G5 H. Y( n/ M% B d0 F" p
给新账户转账
) z ?' e9 i( F( l
我们知道没有余额的账户是没法部署合约的,那我们就从默认账户转1以太币给新账户,使用以下命令(请使用你自己eth.accounts对应输出的账户):
eth.sendTransaction({from:'0xb0ebe17ef0e96b5c525709c0a1ede347c66bd391',to:'0xf280facfd60d61f6fd3f88c9dee4fb90d0e11dfc',value:web3.toWei(1,"ether")})7 v. v8 j" y: J I- C/ `
在打开的tail-ftest.log日志终端里,可以同时看到挖矿记录. U0 C9 D3 R9 n- b6 ~! v" k
# ?" }. b% |* a a; B7 E; {
再次查看新账户余额,可以新账户有1个以太币" K) ^. ], Z& w: f1 u; |
9 \, O) b7 Y P
解锁账户5 p" m4 L& m& c" R% q
在部署合约前需要先解锁账户(就像银行转账要输入密码一样),使用以下命令:, J1 t, \9 w" {7 q5 E S
! i/ q' ^4 Z ], g) }) e8 C" q
personal.unlockAccount(eth.accounts[1],"TinyXiong");. B7 X$ Z. M& O" i
“TinyXiong”是之前创建账户时的密码, t' u0 ]" r: g9 Q |
8 b1 D! A k/ D v) t! I3 `
解锁成功后,账户就准备完毕啦,接下来就是编写合约代码。6 N* {4 Q& w( R% M+ E( C
1 f2 e: X# K4 X, {' X+ H
编写合约代码
! g/ B' c6 h4 Y/ S$ {, n6 E' y
现在我们来开始编写第一个智能合约代码,solidity代码如下:
" m& Z0 D4 `0 E3 j
pragmasolidity^0.4.18;
) r$ D3 H. A0 C* ?0 K
contracthello{9 |& r% ?& C2 m$ i4 ]
stringgreeting;
) ^* m9 X9 l/ b+ f
functionhello(string_greeting)public{) m# y* c! t7 I8 w
' g0 m5 m8 O9 e% z; }- I
greeting=_greeting;
}
functionsay()constantpublicreturns(string){- ?/ _! `7 J, i3 Q& ?
" b: G' t9 n! B( F# H
returngreeting;3 h& U8 ~1 A8 Q/ w
5 y- {" o0 K0 b- H- K- m0 K) k( u
}
}
简单解释下,我们定义了一个名为hello的合约,在合约初始化时保存了一个字符串(我们会传入helloworld),每次调用say返回字符串。8 M i( @3 \/ r# Z
0 e( }/ O, P! Y) k9 _& f
把这段代码写(拷贝)到Browser-Solidity,如果没有错误,点击Details获取部署代码,
如
- i/ ?" X6 G) I- f& z) o9 g! c. X
在弹出的对话框中找到WEB3DEPLOY部分,点拷贝,粘贴到编辑器后,修改初始化字符串为helloworld。. j5 ?! o- p5 e3 K% t: x& g
solidity在博文写作时(2017/11/24),版本为0.4.18,solidity发展非常快,solidity版本之间有可能不能兼容,这是你可以在Browser-Solidity的Settings里选择对应的编译器版本。
2 V+ e2 N5 B9 p/ i% {% ?, z
Browser-Solidity也不停的更新中,截图可能和你看到的界面不一样。
" k7 X2 T# a+ J2 q% Y% s& L
部署合约
Browser-Solidity生成的代码,拷贝到编辑器里修改后的代码如下
6 Z1 {: e/ ~2 [/ s
var_greeting="HelloWorld";7 A9 m- P4 r$ `9 K, y9 F) D* [/ u
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"}]);
varhello=helloContract.new(
_greeting,; ?7 y4 F5 o+ j6 V& [; u
{8 W' ]2 i8 j, G# F
) b1 O2 _: @% W* U, \
from:web3.eth.accounts[1],& o/ K! b; z1 A/ _
4 S8 z- I. ~7 g A6 I8 K; a9 q! C$ K
data:'0x6060604052341561000f57600080fd5b6040516102b83803806102b8833981016040528080518201919050508060009080519060200190610041929190610048565b50506100ed565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061008957805160ff19168380011785556100b7565b828001600101855582156100b7579182015b828111156100b657825182559160200191906001019061009b565b5b5090506100c491906100c8565b5090565b6100ea91905b808211156100e65760008160009055506001016100ce565b5090565b90565b6101bc806100fc6000396000f300606060405260043610610041576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063954ab4b214610046575b600080fd5b341561005157600080fd5b6100596100d4565b6040518080602001828103825283818151815260200191508051906020019080838360005b8381101561009957808201518184015260208101905061007e565b50505050905090810190601f1680156100c65780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6100dc61017c565b60008054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156101725780601f1061014757610100808354040283529160200191610172565b820191906000526020600020905b81548152906001019060200180831161015557829003601f168201915b5050505050905090565b6020604051908101604052806000815250905600a165627a7a723058204a5577bb3ad30e02f7a3bdd90eedcc682700d67fc8ed6604d38bb739c0655df90029',5 `4 C( H# n$ u6 B( K3 d' p
gas:'4700000'- e; J1 m/ D5 z! I' I
},function(e,contract){
; q& ~4 o4 n3 I
console.log(e,contract);
' e8 x2 w( t9 V' U/ u# b
if(typeofcontract.address!=='undefined'){! c7 {, U( @. |& h3 p7 g
console.log('Contractmined!address:'+contract.address+'transactionHash:'+contract.transactionHash);9 L! Z8 @- j" o* R9 u6 L5 b
}
});# U" T/ s$ O- H6 x
第1行:修改字符串为HelloWorld
第2行:修改合约变量名
第3行:修改合约实例变量名,之后可以直接用实例调用函数。
第6行:修改部署账户为新账户索引,即使用新账户来部署合约。2 r8 y0 C4 q1 M# g
第8行:准备付的gas费用,IDE已经帮我们预估好了。
( D- t0 V% Z; ~# }5 V5 e* V
第9行:设置部署回调函数。3 |: j6 i; B1 I+ g
拷贝回geth控制台里,回车后,看到输出如:) c' D4 r5 ~( A
Contractmined!address:0x79544078dcd9d560ec3f6eff0af42a9fc84c7d19transactionHash:0xe2caab22102e93434888a0b8013a7ae7e804b132e4a8bfd2318356f6cf0480b3# r5 u& n9 K }
2 e, D' g" ^3 z: p& \
说明合约已经部署成功。
在打开的tail-ftest.log日志终端里,可以同时看到挖矿记录
现在我们查看下新账户的余额:
>eth.getBalance(eth.accounts[1])! Z9 l0 K. b& _- d
是不是比之前转账的余额少呀!
) O2 I6 T; G. _* W% }
运行合约
>hello.say()8 Q$ ^( n! q0 u& o& A
"HelloWorld", k3 {% i+ w& e' m
输出HelloWorld,我们第一个合约HelloWorld,成功运行了。
运行截图如下:
第一个合约的意义更重要的是体验智能合约开发流程,对于初学者一些可以选择先放弃一些细节,开发流程打通之后,可以增强信心进行下一步的学习。
作者:@xilibi2003
成为第一个吐槽的人