智能合约开发环境搭建及Hello World合约
945坏男人
发表于 2022-11-7 00:10:06
244
0
0
. G1 [" U: c6 Z
就先学习任何编程语言一样,入门的第一个程序都是HelloWorld。今天我们来一步一步从搭建以太坊智能合约开发环境开始,讲解智能合约的HelloWorld如何编写。
开发环境搭建$ a) H- S5 H* p8 {
. H, w0 t! m9 H$ @ Q
Solidity安装$ |3 w$ g" y5 |, |; B
2 L' ~+ V8 v, Z1 L, d) I2 f
强烈建议新手使用Remix-SolidityIDE来进行开发。, ?8 o4 f" [; I9 r( K% c- |
8 r) \$ r; ~3 W4 Q$ W7 u) e1 F. _
Remix是一个基于浏览器的Solidity,就可以不用安装Solidity,本文的HelloWorld教程也将基于RemixSolidityIDE来进行。
& C6 n. O' n, P
如果你想自己安装请参考Solidity安装指引。2 z7 ^+ c1 f- C
* K V' Y( [1 Q0 c8 W
更新,开发环境搭建还可以看另一篇文章:搭建智能合约开发环境RemixIDE及使用。
3 y4 l8 o+ v# A: E3 W
geth安装
7 _5 C5 d* d5 g
Mac下安装命令如下:其他平台参考:geth官方安装指引
5 O1 j$ h+ X1 A+ |0 h: R
brewtapethereum/ethereum2 C. v8 j, N o2 Z B! s" P! V
# x* r* ~6 L* {9 J* Z( R i
brewinstallethereum
3 U' ^! h* K2 t% a) B0 K
brew是Mac下的包管理工具,和Ubuntu里的apt-get类似" O/ ^: u# @7 p
( W* N7 ]1 P8 O8 C
安装完以后,就是把geth控制台启动。
启动环境
在入门篇讲过,geth是一个以太坊客户端,现在利用geth启动一个以太坊(开发者)网络节点。
( x( p: I y' [/ A" w
geth--datadirtestNet--devconsole2>>test.log, D# t4 \7 p$ k, r/ e
执行命名后,会进入geth控制台,这时光标停在一个向右的箭头处,像这样9 W+ ^1 e/ H) |6 _ i
' n' M3 |& @) p8 x" ]/ r: m" N
命令参数说明7 y7 s. W# k$ S' Y0 o" y" C
* F' U: s2 T% e8 F
–dev启用开发者网络(模式),开发者网络会使用POA共识,默认预分配一个开发者账户并且会自动开启挖矿。
–datadir后面的参数是区块数据及秘钥存放目录。4 \* `0 G- _! @$ G# U+ C
第一次输入命令后,它会放在当前目录下新建一个testNet目录来存放数据。' P' Q; |! g; M8 L
console进入控制台
2 p' }0 b( y- J8 H1 s
2>>test.log表示把控制台日志输出到test.log文件1 z! Q3 @! I: ~* J9 ^+ Z
) J/ H; i, r x) Q' E+ ^ T. L
为了更好的理解,建议新开一个命令行终端,实时显示日志:$ j w3 W3 U! `
) P! [0 d7 J, ]7 [7 Q1 s J
tail-ftest.log$ Z, |5 }$ g5 Q4 v: @5 @: {0 I
准备账户& r) {; M) ^. Z! D
部署智能合约需要一个外部账户,我们先来看看分配的开发者账户,在控制台使用以下命令查看账户:* ?3 A6 a: N( d1 \
) E9 ?3 N( j1 `# b/ R1 \3 l4 F
>eth.accounts
回车后,返回一个账户数组,里面有一个默认账户,如:
* W2 t& V" E7 E, o7 Z; b; A
也可以使用personal.listAccounts查看账户,
再来看一下账户里的余额,使用一下命令:0 j* e6 O7 [- c( l; r4 x2 J5 r
1 O; S- _, d3 ^! d' [ ]1 y
>eth.getBalance(eth.accounts[0])
**eth.accounts[0]**表示账户列表第一个账户
2 J( W0 _$ {7 j$ x: [0 @2 y1 G* D
回车后,可以看到大量的余额,如: d ? o. O* Q" s, V
; h6 U# Z d6 J; k8 d
1.15792089237316195423570985008687907853269…e+77
开发者账户因余额太多,如果用这个账户来部署合约时会无法看到余额变化,为了更好的体验完整的过程,这里选择创建一个新的账户。
1 `% E3 W, d1 q3 F& a' y
创建账户
2 f, w. u7 q9 o" @ d$ @
使用以下命令创建账户:
>personal.newAccount("TinyXiong")
4 }1 ~- q% O, U
TinyXiong为新账户的密码,回车后,返回一个新账户。
这时我们查看账户列表:8 b' W7 B7 l/ X+ ^4 n2 x
, v7 z8 Q8 w; {9 c
>eth.accounts+ y+ ~- t+ R. O4 u9 B! Y/ V7 g
0 O0 h) U, E+ c+ w3 y3 ?9 y& D9 v$ B
可以看到账户数组你包含两个账户,新账户在第二个(索引为1)位置。+ Y% g1 Y/ F9 ?5 E
现在看看账户的余额:
5 V+ l0 w2 w4 n7 L
>eth.getBalance(eth.accounts[1])
05 S2 }8 W3 _, X
* D( [9 ?; b s$ s% [/ F$ @
回车后,返回的是0,新账户是0。结果如:
给新账户转账 v5 j8 D+ ?# K4 u" n
) v$ |6 ] g$ P
我们知道没有余额的账户是没法部署合约的,那我们就从默认账户转1以太币给新账户,使用以下命令(请使用你自己eth.accounts对应输出的账户):
& v6 W& B* Y. |( u6 I
eth.sendTransaction({from:'0xb0ebe17ef0e96b5c525709c0a1ede347c66bd391',to:'0xf280facfd60d61f6fd3f88c9dee4fb90d0e11dfc',value:web3.toWei(1,"ether")})
" A- `& m9 h+ i+ [! {) B2 L( L
在打开的tail-ftest.log日志终端里,可以同时看到挖矿记录. [6 Z p- q" C" N2 i! n8 T4 r
再次查看新账户余额,可以新账户有1个以太币
解锁账户9 T* A+ o$ [5 J0 m
) ]! U; ]2 z, C. D
在部署合约前需要先解锁账户(就像银行转账要输入密码一样),使用以下命令:
9 N+ j; D& L( K. \9 K& I
personal.unlockAccount(eth.accounts[1],"TinyXiong");" u7 W2 R; b( V: b$ r6 y
! ^8 P3 v; u7 M1 L+ O
“TinyXiong”是之前创建账户时的密码7 |. Z0 T8 ?9 P; H6 V$ h" T+ a
解锁成功后,账户就准备完毕啦,接下来就是编写合约代码。
编写合约代码
现在我们来开始编写第一个智能合约代码,solidity代码如下:
pragmasolidity^0.4.18;3 B5 t/ k6 f9 y' }! L; g
contracthello{
; M2 ~2 |, c6 p
stringgreeting;
2 J" s) O! m% M5 [8 [8 @ T
functionhello(string_greeting)public{) v% l" ]/ V* ^: i/ z" l) X
8 ]8 |1 |! J" w: T
greeting=_greeting;7 F; X6 _& e/ i5 u! B
}' ^8 z' ~# l! S `+ L4 v
0 J, z4 j% c0 D$ ?0 W' G7 p2 x
functionsay()constantpublicreturns(string){
, E2 y* h7 g9 B' r/ ^
returngreeting;. c1 E) X6 L j+ g! ^' [0 E9 B4 k
: N' ^3 E7 p e2 e" _! u) _( m7 J! Y
}
; U0 e) H/ n) Y/ C8 r
}. e+ ^& K5 U* H# p3 t8 |9 P1 j
简单解释下,我们定义了一个名为hello的合约,在合约初始化时保存了一个字符串(我们会传入helloworld),每次调用say返回字符串。% W+ t3 R) Q: t' D
把这段代码写(拷贝)到Browser-Solidity,如果没有错误,点击Details获取部署代码,
5 v0 Q' V7 k( e0 N9 i! D
如
在弹出的对话框中找到WEB3DEPLOY部分,点拷贝,粘贴到编辑器后,修改初始化字符串为helloworld。7 ]% Z x) ]" f! n- {7 d
solidity在博文写作时(2017/11/24),版本为0.4.18,solidity发展非常快,solidity版本之间有可能不能兼容,这是你可以在Browser-Solidity的Settings里选择对应的编译器版本。
3 D& @6 M: A6 I2 t# P+ P, v [( @" F! J
Browser-Solidity也不停的更新中,截图可能和你看到的界面不一样。5 Y6 ]8 V( u5 ]1 t1 \
部署合约9 a9 ?1 {8 D7 U& {; g6 b5 \) B# |
/ T" o8 n9 g) N) S' }, n) f
Browser-Solidity生成的代码,拷贝到编辑器里修改后的代码如下
! N- Q* d, M0 W
var_greeting="HelloWorld";
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(& j! Q0 |# M1 z, y I
6 J, {: g: f0 x w* n4 v0 u
_greeting,
{
8 F7 k. I- |; B' B
from:web3.eth.accounts[1],
0 \: h' Q, w9 p$ n$ g& z7 P& _
data:'0x6060604052341561000f57600080fd5b6040516102b83803806102b8833981016040528080518201919050508060009080519060200190610041929190610048565b50506100ed565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061008957805160ff19168380011785556100b7565b828001600101855582156100b7579182015b828111156100b657825182559160200191906001019061009b565b5b5090506100c491906100c8565b5090565b6100ea91905b808211156100e65760008160009055506001016100ce565b5090565b90565b6101bc806100fc6000396000f300606060405260043610610041576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063954ab4b214610046575b600080fd5b341561005157600080fd5b6100596100d4565b6040518080602001828103825283818151815260200191508051906020019080838360005b8381101561009957808201518184015260208101905061007e565b50505050905090810190601f1680156100c65780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6100dc61017c565b60008054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156101725780601f1061014757610100808354040283529160200191610172565b820191906000526020600020905b81548152906001019060200180831161015557829003601f168201915b5050505050905090565b6020604051908101604052806000815250905600a165627a7a723058204a5577bb3ad30e02f7a3bdd90eedcc682700d67fc8ed6604d38bb739c0655df90029',
gas:'4700000'
},function(e,contract){
console.log(e,contract);
4 I7 Z4 p) Q& G3 ~/ |2 S3 F# {8 v
if(typeofcontract.address!=='undefined'){
console.log('Contractmined!address:'+contract.address+'transactionHash:'+contract.transactionHash);/ b( j3 f$ y: l0 @: g
}
});
第1行:修改字符串为HelloWorld9 ]; z- n3 p' Z# J8 m% ~2 R5 E
第2行:修改合约变量名" ~ ^; X/ k4 a, @: E
第3行:修改合约实例变量名,之后可以直接用实例调用函数。1 B. w1 P+ ~4 t& ?
# w- e! r. `6 C& |3 j& w O
第6行:修改部署账户为新账户索引,即使用新账户来部署合约。
, H% r- S8 ~$ ~- d1 [' j' C' ^
第8行:准备付的gas费用,IDE已经帮我们预估好了。* l$ Y, ~1 a2 e: ?3 x4 h
第9行:设置部署回调函数。
拷贝回geth控制台里,回车后,看到输出如:- q0 s5 u% M$ r! y9 t1 B3 o% A& F, F
* N# J, p+ f- z) I
Contractmined!address:0x79544078dcd9d560ec3f6eff0af42a9fc84c7d19transactionHash:0xe2caab22102e93434888a0b8013a7ae7e804b132e4a8bfd2318356f6cf0480b3+ y* S9 W) ?; i% Y3 e d' u' j
, h7 b$ e+ Q$ a2 I0 `7 e+ k/ p
说明合约已经部署成功。- S f1 S* k# L: c# s L
/ \" j- _/ n1 r0 b% S
在打开的tail-ftest.log日志终端里,可以同时看到挖矿记录6 F( A8 v5 ?- p ^6 w8 O3 [
% I, O, ?; o' h% t- u) ^
现在我们查看下新账户的余额:
>eth.getBalance(eth.accounts[1])3 B+ }1 k [5 y- b1 p( g% W& [9 |
是不是比之前转账的余额少呀!7 C J# f) O5 j
运行合约
>hello.say()( C! q/ y! W' ~; j
"HelloWorld"' R+ Y" T6 k. z
输出HelloWorld,我们第一个合约HelloWorld,成功运行了。0 w7 @ S* z5 c4 P: y# G
- ^( g- z! H' \: @
运行截图如下:
第一个合约的意义更重要的是体验智能合约开发流程,对于初学者一些可以选择先放弃一些细节,开发流程打通之后,可以增强信心进行下一步的学习。
作者:@xilibi2003
成为第一个吐槽的人