教程 | 以太坊智能合约编程之菜鸟教程
教程 | 以太坊智能合約編程之菜鳥教程
有些人說以太坊太難對(duì)付,于是我們(譯注:指Consensys, 下同)寫了這篇文章來幫助大家學(xué)習(xí)如何利用以太坊編寫智能合約和應(yīng)用。這里所用到的工具,錢包,應(yīng)用程序以及整個(gè)生態(tài)系統(tǒng)仍處于開發(fā)狀態(tài),它們將來會(huì)更好用!
- 第一部分概述,討論了關(guān)鍵概念,幾大以太坊客戶端以及寫智能合約用到的編程語言。
- 第二部分討論了總體的工作流程,以及目前流行的一些DApp框架和工具。
- 第三部分主要關(guān)于編程,我們將學(xué)習(xí)如何使用Truffle來為智能合約編寫測(cè)試和構(gòu)建DApp。
第一部分. 概述
如果你對(duì)諸如比特幣以及其工作原理等密碼學(xué)貨幣的概念完全陌生,我們建議你先看看Andreas Antonopoulos所著的Bitcoin Book的頭幾章,然后讀一下以太坊白皮書。(譯注:以太坊白皮書中文版請(qǐng)看?http://ethfans.org/posts/ethereum-whitepaper)
如果你覺得白皮書中的章節(jié)太晦澀,也可以直接動(dòng)手來熟悉以太坊。在以太坊上做開發(fā)并不要求你理解所有那些“密碼經(jīng)濟(jì)計(jì)算機(jī)科學(xué)”(crypto economic computer science),而白皮書的大部分是關(guān)于以太坊想對(duì)于比特幣架構(gòu)上的改進(jìn)。
新手教程
ethereum.org提供了官方的新手入門教程,以及一個(gè)代幣合約和眾籌合約的教程。合約語言Solidity也有官方文檔。學(xué)習(xí)智能合約的另一份不錯(cuò)的資料(也是我的入門資料)是dappsForBeginners,不過現(xiàn)在可能有些過時(shí)了。
這篇文章的目的是成為上述資料的補(bǔ)充,同時(shí)介紹一些基本的開發(fā)者工具,使入門以太坊,智能合約以及構(gòu)建DApps(decentralized apps, 分布式應(yīng)用)更加容易。我會(huì)試圖按照我自己(依然是新手)的理解來解釋工作流程中的每一步是在做什么,我也得到了ConsenSys酷酷的開發(fā)者們的許多幫助。
基本概念
了解這些名詞是一個(gè)不錯(cuò)的開始:
公鑰加密系統(tǒng)。?Alice有一把公鑰和一把私鑰。她可以用她的私鑰創(chuàng)建數(shù)字簽名,而Bob可以用她的公鑰來驗(yàn)證這個(gè)簽名確實(shí)是用Alice的私鑰創(chuàng)建的,也就是說,確實(shí)是Alice的簽名。當(dāng)你創(chuàng)建一個(gè)以太坊或者比特幣錢包的時(shí)候,那長(zhǎng)長(zhǎng)的0xdf...5f地址實(shí)質(zhì)上是個(gè)公鑰,對(duì)應(yīng)的私鑰保存某處。類似于Coinbase的在線錢包可以幫你保管私鑰,你也可以自己保管。如果你弄丟了存有資金的錢包的私鑰,你就等于永遠(yuǎn)失去了那筆資金,因此你最好對(duì)私鑰做好備份。過來人表示:通過踩坑學(xué)習(xí)到這一點(diǎn)是非常痛苦的...
點(diǎn)對(duì)點(diǎn)網(wǎng)絡(luò)。?就像BitTorrent, 以太坊分布式網(wǎng)絡(luò)中的所有節(jié)點(diǎn)都地位平等,沒有中心服務(wù)器。(未來會(huì)有半中心化的混合型服務(wù)出現(xiàn)為用戶和開發(fā)者提供方便,這我們后面會(huì)講到。)
區(qū)塊鏈。?區(qū)塊鏈就像是一個(gè)全球唯一的帳簿,或者說是數(shù)據(jù)庫,記錄了網(wǎng)絡(luò)中所有交易歷史。
以太坊虛擬機(jī)(EVM)。?它讓你能在以太坊上寫出更強(qiáng)大的程序(比特幣上也可以寫腳本程序)。它有時(shí)也用來指以太坊區(qū)塊鏈,負(fù)責(zé)執(zhí)行智能合約以及一切。
節(jié)點(diǎn)。?你可以運(yùn)行節(jié)點(diǎn),通過它讀寫以太坊區(qū)塊鏈,也即使用以太坊虛擬機(jī)。完全節(jié)點(diǎn)需要下載整個(gè)區(qū)塊鏈。輕節(jié)點(diǎn)仍在開發(fā)中。
礦工。?挖礦,也就是處理區(qū)塊鏈上的區(qū)塊的節(jié)點(diǎn)。這個(gè)網(wǎng)頁可以看到當(dāng)前活躍的一部分以太坊礦工:stats.ethdev.com。
工作量證明。?礦工們總是在競(jìng)爭(zhēng)解決一些數(shù)學(xué)問題。第一個(gè)解出答案的(算出下一個(gè)區(qū)塊)將獲得以太幣作為獎(jiǎng)勵(lì)。然后所有節(jié)點(diǎn)都更新自己的區(qū)塊鏈。所有想要算出下一個(gè)區(qū)塊的礦工都有與其他節(jié)點(diǎn)保持同步,并且維護(hù)同一個(gè)區(qū)塊鏈的動(dòng)力,因此整個(gè)網(wǎng)絡(luò)總是能達(dá)成共識(shí)。(注意:以太坊正計(jì)劃轉(zhuǎn)向沒有礦工的權(quán)益證明系統(tǒng)(POS),不過那不在本文討論范圍之內(nèi)。)
以太幣。?縮寫ETH。一種你可以購買和使用的真正的數(shù)字貨幣。這里是可以交易以太幣的其中一家交易所的走勢(shì)圖。在寫這篇文章的時(shí)候,1個(gè)以太幣價(jià)值65美分。
Gas. (汽油)?在以太坊上執(zhí)行程序以及保存數(shù)據(jù)都要消耗一定量的以太幣,Gas是以太幣轉(zhuǎn)換而成。這個(gè)機(jī)制用來保證效率。
DApp.?以太坊社區(qū)把基于智能合約的應(yīng)用稱為去中心化的應(yīng)用程序(Decentralized App)。DApp的目標(biāo)是(或者應(yīng)該是)讓你的智能合約有一個(gè)友好的界面,外加一些額外的東西,例如IPFS(可以存儲(chǔ)和讀取數(shù)據(jù)的去中心化網(wǎng)絡(luò),不是出自以太坊團(tuán)隊(duì)但有類似的精神)。DApp可以跑在一臺(tái)能與以太坊節(jié)點(diǎn)交互的中心化服務(wù)器上,也可以跑在任意一個(gè)以太坊平等節(jié)點(diǎn)上。(花一分鐘思考一下:與一般的網(wǎng)站不同,DApp不能跑在普通的服務(wù)器上。他們需要提交交易到區(qū)塊鏈并且從區(qū)塊鏈而不是中心化數(shù)據(jù)庫讀取重要數(shù)據(jù)。相對(duì)于典型的用戶登錄系統(tǒng),用戶有可能被表示成一個(gè)錢包地址而其它用戶數(shù)據(jù)保存在本地。許多事情都會(huì)與目前的web應(yīng)用有不同架構(gòu)。)
如果想看看從另一個(gè)新手視角怎么理解這些概念,請(qǐng)讀Just Enough Bitcoin for Ethereum。
以太坊客戶端,智能合約語言
編寫和部署智能合約并不要求你運(yùn)行一個(gè)以太坊節(jié)點(diǎn)。下面有列出基于瀏覽器的IDE和API。但如果是為了學(xué)習(xí)的話,還是應(yīng)該運(yùn)行一個(gè)以太坊節(jié)點(diǎn),以便理解其中的基本組件,何況運(yùn)行節(jié)點(diǎn)也不難。
運(yùn)行以太坊節(jié)點(diǎn)可用的客戶端
以太坊有許多不同語言的客戶端實(shí)現(xiàn)(即多種與以太坊網(wǎng)絡(luò)交互的方法),包括C++, Go, Python, Java, Haskell等等。為什么需要這么多實(shí)現(xiàn)?不同的實(shí)現(xiàn)能滿足不同的需求(例如Haskell實(shí)現(xiàn)的目標(biāo)是可以被數(shù)學(xué)驗(yàn)證),能使以太坊更加安全,能豐富整個(gè)生態(tài)系統(tǒng)。
在寫作本文時(shí),我使用的是Go語言實(shí)現(xiàn)的客戶端geth (go-ethereum),其他時(shí)候還會(huì)使用一個(gè)叫testrpc的工具, 它使用了Python客戶端pyethereum。后面的例子會(huì)用到這些工具。
注: 我曾經(jīng)使用過C++的客戶端,現(xiàn)在仍然在用其中的ethminer組件和geth配合挖礦,因此這些不同的組件是可以一起工作的。
關(guān)于挖礦:挖礦很有趣,有點(diǎn)像精心照料你的室內(nèi)盆栽,同時(shí)又是一種了解整個(gè)系統(tǒng)的方法。雖然以太幣現(xiàn)在的價(jià)格可能連電費(fèi)都補(bǔ)不齊,但以后誰知道呢。人們正在創(chuàng)造許多酷酷的DApp, 可能會(huì)讓以太坊越來越流行。
交互式控制臺(tái)。?客戶端運(yùn)行起來后,你就可以同步區(qū)塊鏈,建立錢包,收發(fā)以太幣了。使用geth的一種方式是通過Javascript控制臺(tái)(JavaScript console, 類似你在chrome瀏覽器里面按F12出來的那個(gè),只不過是跑在終端里)。此外還可以使用類似cURL的命令通過JSON RPC來與客戶端交互。本文的目標(biāo)是帶大家過一邊DApp開發(fā)的流程,因此這塊就不多說了。但是我們應(yīng)該記住這些命令行工具是調(diào)試,配置節(jié)點(diǎn),以及使用錢包的利器。
在測(cè)試網(wǎng)絡(luò)運(yùn)行節(jié)點(diǎn)。?如果你在正式網(wǎng)絡(luò)運(yùn)行g(shù)eth客戶端,下載整個(gè)區(qū)塊鏈與網(wǎng)絡(luò)同步會(huì)需要相當(dāng)時(shí)間。(你可以通過比較節(jié)點(diǎn)日志中打印的最后一個(gè)塊號(hào)和stats.ethdev.com上列出的最新塊來確定是否已經(jīng)同步。) 另一個(gè)問題是在正式網(wǎng)絡(luò)上跑智能合約需要實(shí)實(shí)在在的以太幣。在測(cè)試網(wǎng)絡(luò)上運(yùn)行節(jié)點(diǎn)的話就沒有這個(gè)問題。此時(shí)也不需要同步整個(gè)區(qū)塊鏈,創(chuàng)建一個(gè)自己的私有鏈就勾了,對(duì)于開發(fā)來說更省時(shí)間。
testrpc.?用geth可以創(chuàng)建一個(gè)測(cè)試網(wǎng)絡(luò),另一種更快的創(chuàng)建測(cè)試網(wǎng)絡(luò)的方法是使用testrpc. Testrpc可以在啟動(dòng)時(shí)幫你創(chuàng)建一堆存有資金的測(cè)試賬戶。它的運(yùn)行速度也更快因此更適合開發(fā)和測(cè)試。你可以從testrpc起步,然后隨著合約慢慢成型,轉(zhuǎn)移到geth創(chuàng)建的測(cè)試網(wǎng)絡(luò)上 - 啟動(dòng)方法很簡(jiǎn)單,只需要指定一個(gè)networkid:geth --networkid "12345"。這里是testrpc的代碼倉庫,下文我們還會(huì)再講到它。
接下來我們來談?wù)効捎玫木幊陶Z言,之后就可以開始真正的編程了。
寫智能合約用的編程語言
用Solidity就好。?要寫智能合約有好幾種語言可選:有點(diǎn)類似Javascript的Solidity, 文件擴(kuò)展名是.sol. 和Python接近的Serpent, 文件名以.se結(jié)尾。還有類似Lisp的LLL。Serpent曾經(jīng)流行過一段時(shí)間,但現(xiàn)在最流行而且最穩(wěn)定的要算是Solidity了,因此用Solidity就好。聽說你喜歡Python? 用Solidity。
solc編譯器。?用Solidity寫好智能合約之后,需要用solc來編譯。它是一個(gè)來自C++客戶端實(shí)現(xiàn)的組件(又一次,不同的實(shí)現(xiàn)產(chǎn)生互補(bǔ)),這里是安裝方法。如果你不想安裝solc也可以直接使用基于瀏覽器的編譯器,例如Solidity real-time compiler或者Cosmo。后文有關(guān)編程的部分會(huì)假設(shè)你安裝了solc。
注意:以太坊正處于積極的開發(fā)中,有時(shí)候新的版本之間會(huì)有不同步。確認(rèn)你使用的是最新的dev版本,或者穩(wěn)定版本。如果遇到問題可以去以太坊項(xiàng)目對(duì)應(yīng)的Gitter聊天室或者forums.ethereum.org上問問其他人在用什么版本。
web3.js API.?當(dāng)Solidity合約編譯好并且發(fā)送到網(wǎng)絡(luò)上之后,你可以使用以太坊的web3.js JavaScript API來調(diào)用它,構(gòu)建能與之交互的web應(yīng)用。
以上就是在以太坊上編寫智能合約和構(gòu)建與之交互的DApp所需的基本工具。
第二部分. DApp框架,工具以及工作流程
DApp開發(fā)框架
雖然有上文提到的工具就可以進(jìn)行開發(fā)了,但是使用社區(qū)大神們創(chuàng)造的框架會(huì)讓開發(fā)更容易。
Truffle and Embark.?是Truffle把我領(lǐng)進(jìn)了門。在Truffle出現(xiàn)之前的那個(gè)夏天,我目睹了一幫有天分的學(xué)生是如何不眠不休的參加一個(gè)hackathon(編程馬拉松)活動(dòng)的,雖然結(jié)果相當(dāng)不錯(cuò),但我還是嚇到了。然后Truffle出現(xiàn)了,幫你處理掉大量無關(guān)緊要的小事情,讓你可以迅速進(jìn)入寫代碼-編譯-部署-測(cè)試-打包DApp這個(gè)流程。另外一個(gè)相似的DApp構(gòu)建與測(cè)試框架是Embark。我只用過Truffle, 但是兩個(gè)陣營(yíng)都擁有不少DApp大神。
Meteor.?許多DApp開發(fā)者使用的另一套開發(fā)棧由web3.js和Meteor組成,Meteor是一套通用webapp開發(fā)框架(ethereum-meteor-wallet項(xiàng)目提供了一個(gè)很棒的入門實(shí)例,而SilentCiero正在構(gòu)建大量Meteor與web3.js和DApp集成的模板)。我下載并運(yùn)行過一些不錯(cuò)的DApp是以這種方式構(gòu)造的。在11月9日至13日的以太坊開發(fā)者大會(huì)DΞVCON1上將有一些有趣的討論,是關(guān)于使用這些工具構(gòu)建DApp以及相關(guān)最佳實(shí)踐的(會(huì)議將會(huì)在YouTube上直播)。
APIs.?BlockApps.net打算提供一套R(shí)ESTful API給DApp使用以免去開發(fā)者運(yùn)行本地節(jié)點(diǎn)的麻煩,這個(gè)中心化服務(wù)是基于以太坊Haskell實(shí)現(xiàn)的。這與DApp的去中心化模型背道而馳,但是在本地?zé)o法運(yùn)行以太坊節(jié)點(diǎn)的場(chǎng)合非常有用,比如在你希望只有瀏覽器或者使用移動(dòng)設(shè)備的用戶也能使用你的DApp的時(shí)候。BlockApps提供了一個(gè)命令行工具bloc,注冊(cè)一個(gè)開發(fā)者帳號(hào)之后就可以使用。
許多人擔(dān)心需要運(yùn)行以太坊節(jié)點(diǎn)才能使用DApp的話會(huì)把用戶嚇跑,其實(shí)包括BlockApps在內(nèi)的許多工具都能解決這個(gè)問題。Metamask允許你在瀏覽器里面使用以太坊的功能而無需節(jié)點(diǎn),以太坊官方提供的AlethZero或者AlethOne是正在開發(fā)中有易用界面的客戶端,ConsenSys正在打造一個(gè)輕錢包LightWallet,這些工具都會(huì)讓DApp的使用變得更容易。輕客戶端和水平分片(sharding)也在計(jì)劃和開發(fā)之中。這是一個(gè)能進(jìn)化出混合架構(gòu)的P2P生態(tài)系統(tǒng)。
智能合約集成開發(fā)環(huán)境 (IDE)
IDE.?以太坊官方出品了用來編寫智能合約的Mix IDE,我還沒用過但會(huì)盡快一試。
基于瀏覽器的IDE.?Solidity real-time compiler和Cosmo都可以讓你快速開始在瀏覽器中編寫智能合約。你甚至可以讓這些工具使用你的本地節(jié)點(diǎn),只要讓本地節(jié)點(diǎn)開一個(gè)端口(注意安全!這些工具站點(diǎn)必須可信,而且千萬不要把你的全部身家放在這樣一個(gè)本地節(jié)點(diǎn)里面!Cosmo UI上有如何使用geth做到這一點(diǎn)的指引)。在你的智能合約調(diào)試通過之后,可以用開發(fā)框架來給它添加用戶界面和打包成DApp,這正是Truffle的工作,后面的編程章節(jié)會(huì)有詳細(xì)講解。
Ether.Camp正在開發(fā)另一個(gè)強(qiáng)大的企業(yè)級(jí)瀏覽器IDE。他們的IDE將支持沙盒測(cè)試網(wǎng)絡(luò),自動(dòng)生成用于測(cè)試的用戶界面(取代后文將展示的手動(dòng)編寫測(cè)試),以及一個(gè)測(cè)試交易瀏覽器test.ether.camp。當(dāng)你的合約準(zhǔn)備正式上線之前,使用他們的測(cè)試網(wǎng)絡(luò)會(huì)是確保你的智能合約在一個(gè)接近真實(shí)的環(huán)境工作正常的好方法。他們也為正式網(wǎng)絡(luò)提供了一個(gè)交易瀏覽器frontier.ether.camp,上面可以看到每一筆交易的細(xì)節(jié)。在本文寫作時(shí)Ether.Camp的IDE還只能通過邀請(qǐng)注冊(cè),預(yù)計(jì)很快會(huì)正式發(fā)布。
合約和Dapp示例。?在Github上搜索DApp倉庫和.sol文件可以看到進(jìn)行中的有趣東西。這里有一個(gè)DApp大列表:dapps.ethercasts.com,不過其中一些項(xiàng)目已經(jīng)過時(shí)。Ether.fund/contracts上有一些Solidity和Serpent寫的合約示例,但是不清楚這些例子有沒有經(jīng)過測(cè)試或者正確性驗(yàn)證。11月12日的開發(fā)者大會(huì)DΞVCON1將會(huì)有一整天的DApp主題演講。
部署智能合約的流程
流程如下:
下圖詳細(xì)描繪了這個(gè)流程:
你的DApp可以給用戶提供一個(gè)界面先部署所需合約再使用之(如圖1到4步),也可以假設(shè)合約已經(jīng)部署了(常見方法),直接從使用合約(如圖第6步)的界面開始。
第三部分. 編程
在Truffle中進(jìn)行測(cè)試
Truffle用來做智能合約的測(cè)試驅(qū)動(dòng)開發(fā)(TDD)非常棒,我強(qiáng)烈推薦你在學(xué)習(xí)中使用它。它也是學(xué)習(xí)使用JavaScript Promise的一個(gè)好途徑,例如deferred和異步調(diào)用。Promise機(jī)制有點(diǎn)像是說“做這件事,如果結(jié)果是這樣,做甲,如果結(jié)果是那樣,做乙... 與此同時(shí)不要在那兒干等著結(jié)果返回,行不?”。Truffle使用了包裝web3.js的一個(gè)JS Promise框架Pudding(因此它為為你安裝web3.js)。(譯注:Promise是流行于JavaScript社區(qū)中的一種異步調(diào)用模式。它很好的封裝了異步調(diào)用,使其能夠靈活組合,而不會(huì)陷入callback hell.)
Transaction times.?Promise對(duì)于DApp非常有用,因?yàn)榻灰讓懭胍蕴粎^(qū)塊鏈需要大約12-15秒的時(shí)間。即使在測(cè)試網(wǎng)絡(luò)上看起來沒有那么慢,在正式網(wǎng)絡(luò)上卻可能會(huì)要更長(zhǎng)的時(shí)間(例如你的交易可能用光了Gas,或者被寫入了一個(gè)孤兒塊)。
下面讓我們給一個(gè)簡(jiǎn)單的智能合約寫測(cè)試用例吧。
使用Truffle
首先確保你 1.安裝好了solc以及 2.testrpc。(testrpc需要Python和pip。如果你是Python新手,你可能需要用virtualenv來安裝,這可以將Python程序庫安裝在一個(gè)獨(dú)立的環(huán)境中。)
接下來安裝 3.Truffle(你可以使用NodeJS's npm來安裝:npm install -g truffle,?-g開關(guān)可能會(huì)需要sudo)。安裝好之后,在命令行中輸入truffle list來驗(yàn)證安裝成功。然后創(chuàng)建一個(gè)新的項(xiàng)目目錄(我把它命名為'conference'),進(jìn)入這個(gè)目錄,運(yùn)行truffle init。該命令會(huì)建立如下的目錄結(jié)構(gòu):
現(xiàn)在讓我們?cè)诹硪粋€(gè)終端里通過執(zhí)行testrpc來啟動(dòng)一個(gè)節(jié)點(diǎn)(你也可以用geth):
回到之前的終端中,輸入truffle deploy。這條命令會(huì)部署之前truffle init產(chǎn)生的模板合約到網(wǎng)絡(luò)上。任何你可能遇到的錯(cuò)誤信息都會(huì)在testrpc的終端或者執(zhí)行truffle的終端中輸出。
在開發(fā)過程中你隨時(shí)可以使用truffle compile命令來確認(rèn)你的合約可以正常編譯(或者使用solc YourContract.sol),truffle deploy來編譯和部署合約,最后是truffle test來運(yùn)行智能合約的測(cè)試用例。
第一個(gè)合約
下面是一個(gè)針對(duì)會(huì)議的智能合約,通過它參會(huì)者可以買票,組織者可以設(shè)置參會(huì)人數(shù)上限,以及退款策略。本文涉及的所有代碼都可以在這個(gè)代碼倉庫找到。
contract Conference {address public organizer;mapping (address => uint) public registrantsPaid;uint public numRegistrants;uint public quota;event Deposit(address _from, uint _amount); // so you can log these eventsevent Refund(address _to, uint _amount); function Conference() { // Constructororganizer = msg.sender;quota = 500;numRegistrants = 0;}function buyTicket() public returns (bool success) {if (numRegistrants >= quota) { return false; }registrantsPaid[msg.sender] = msg.value;numRegistrants++;Deposit(msg.sender, msg.value);return true;}function changeQuota(uint newquota) public {if (msg.sender != organizer) { return; }quota = newquota;}function refundTicket(address recipient, uint amount) public {if (msg.sender != organizer) { return; }if (registrantsPaid[recipient] == amount) { address myAddress = this;if (myAddress.balance >= amount) { recipient.send(amount);registrantsPaid[recipient] = 0;numRegistrants--;Refund(recipient, amount);}}}function destroy() { // so funds not locked in contract foreverif (msg.sender == organizer) { suicide(organizer); // send funds to organizer}} }接下來讓我們部署這個(gè)合約。(注意:本文寫作時(shí)我使用的是Mac OS X 10.10.5, solc 0.1.3+ (通過brew安裝),Truffle v0.2.3, testrpc v0.1.18 (使用venv))
部署合約
(譯注:圖中步驟翻譯如下:)
使用truffle部署智能合約的步驟:
1.?truffle init?(在新目錄中) => 創(chuàng)建truffle項(xiàng)目目錄結(jié)構(gòu)
2. 編寫合約代碼,保存到contracts/YourContractName.sol文件。
3. 把合約名字加到config/app.json的'contracts'部分。
4. 啟動(dòng)以太坊節(jié)點(diǎn)(例如在另一個(gè)終端里面運(yùn)行testrpc)。
5.?truffle deploy(在truffle項(xiàng)目目錄中)
添加一個(gè)智能合約。?在truffle init執(zhí)行后或是一個(gè)現(xiàn)有的項(xiàng)目目錄中,復(fù)制粘帖上面的會(huì)議合約到contracts/Conference.sol文件中。然后打開config/app.json文件,把'Conference'加入'deploy'數(shù)組中。
啟動(dòng)testrpc。?在另一個(gè)終端中啟動(dòng)testrpc。
編譯或部署。?執(zhí)行truffle compile看一下合約是否能成功編譯,或者直接truffle deploy一步完成編譯和部署。這條命令會(huì)把部署好的合約的地址和ABI(應(yīng)用接口)加入到配置文件中,這樣之后的truffle test和truffle build步驟可以使用這些信息。
出錯(cuò)了??編譯是否成功了?記住,錯(cuò)誤信息即可能出現(xiàn)在testrpc終端也可能出現(xiàn)在truffle終端。
重啟節(jié)點(diǎn)后記得重新部署!?如果你停止了testrpc節(jié)點(diǎn),下一次使用任何合約之前切記使用truffle deploy重新部署。testrpc在每一次重啟之后都會(huì)回到完全空白的狀態(tài)。
合約代碼解讀
讓我們從智能合約頭部的變量聲明開始:
address public organizer; mapping (address => uint) public registrantsPaid; uint public numRegistrants; uint public quota;address.?地址類型。第一個(gè)變量是會(huì)議組織者的錢包地址。這個(gè)地址會(huì)在合約的構(gòu)造函數(shù)function Conference()中被賦值。很多時(shí)候也稱呼這種地址為'owner'(所有人)。
uint.?無符號(hào)整型。區(qū)塊鏈上的存儲(chǔ)空間很緊張,保持?jǐn)?shù)據(jù)盡可能的小。
public.?這個(gè)關(guān)鍵字表明變量可以被合約之外的對(duì)象使用。private修飾符則表示變量只能被本合約(或者衍生合約)內(nèi)的對(duì)象使用。如果你想要在測(cè)試中通過web3.js使用合約中的某個(gè)變量,記得把它聲明為public。
Mapping或數(shù)組。(譯注:Mapping類似Hash, Directory等數(shù)據(jù)類型,不做翻譯。)在Solidity加入數(shù)組類型之前,大家都使用類似mapping (address => uint)的Mapping類型。這個(gè)聲明也可以寫作address registrantsPaid[],不過Mapping的存儲(chǔ)占用更小(smaller footprint)。這個(gè)Mapping變量會(huì)用來保存參加者(用他們的錢包地址表示)的付款數(shù)量以便在退款時(shí)使用。
關(guān)于地址。?你的客戶端(比如testrpc或者geth)可以生成一個(gè)或多個(gè)賬戶/地址。testrpc啟動(dòng)時(shí)會(huì)顯示10個(gè)可用地址:
第一個(gè)地址,?accounts[0],是發(fā)起調(diào)用的默認(rèn)地址,如果沒有特別指定的話。
組織者地址 vs. 合約地址。?部署好的合約會(huì)在區(qū)塊鏈上擁有自己的地址(與組織者擁有的是不同的地址)。在Solidity合約中可以使用this來訪問這個(gè)合約地址,正如refundTicket函數(shù)所展示的:address myAddress = this;
Suicide, Solidity的好東西。(譯注:suicide意為'自殺', 為Solidity提供的關(guān)鍵字,不做翻譯。)轉(zhuǎn)給合約的資金會(huì)保存于合約(地址)中。最終這些資金通過destroy函數(shù)被釋放給了構(gòu)造函數(shù)中設(shè)置的組織者地址。這是通過suicide(orgnizer);這行代碼實(shí)現(xiàn)的。沒有這個(gè),資金可能被永遠(yuǎn)鎖定在合約之中(reddit上有些人就遇到過),因此如果你的合約會(huì)接受資金一定要記得在合約中使用這個(gè)方法!
如果想要模擬另一個(gè)用戶或者對(duì)手方(例如你是賣家想要模擬一個(gè)買家),你可以使用可用地址數(shù)組中另外的地址。假設(shè)你要以另一個(gè)用戶,accounts[1], 的身份來買票,可以通過from參數(shù)設(shè)置:
conference.buyTicket({ from: accounts[1], value: some_ticket_price_integer });函數(shù)調(diào)用可以是交易。?改變合約狀態(tài)(修改變量值,添加記錄,等等)的函數(shù)調(diào)用本身也是轉(zhuǎn)賬交易,隱式的包含了發(fā)送人和交易價(jià)值。因此web3.js的函數(shù)調(diào)用可以通過指定{ from: __, value: __ }參數(shù)來發(fā)送以太幣。在Solidity合約中,你可以通過msg.sender和msg.value來獲取這些信息:
function buyTicket() public {...registrantsPaid[msg.sender] = msg.value;... }事件(Event)。?可選的功能。合約中的Deposit(充值)和Send(發(fā)送)事件是會(huì)被記錄在以太坊虛擬機(jī)日志中的數(shù)據(jù)。它們實(shí)際上沒有任何作用,但是用事件(Event)把交易記錄進(jìn)日志是好的做法。
好了,現(xiàn)在讓我們給這個(gè)智能合約寫一個(gè)測(cè)試,來確保它能工作。
寫測(cè)試
把項(xiàng)目目錄test/中的example.js文件重命名為conference.js,文件中所有的'Example'替換為'Conference'。
contract('Conference', function(accounts) {it("should assert true", function(done) {var conference = Conference.at(Conference.deployed_address);assert.isTrue(true);done(); // stops tests at this point}); });在項(xiàng)目根目錄下運(yùn)行truffle test,你應(yīng)該看到測(cè)試通過。在上面的測(cè)試中truffle通過Conference.deployed_address獲得合約部署在區(qū)塊鏈上的地址。
讓我們寫一個(gè)測(cè)試來初始化一個(gè)新的Conference,然后檢查變量都正確賦值了。將conference.js中的測(cè)試代碼替換為:
contract('Conference', function(accounts) {it("Initial conference settings should match", function(done) {var conference = Conference.at(Conference.deployed_address); // same as previous example up to hereConference.new({ from: accounts[0] }).then(function(conference) {conference.quota.call().then(function(quota) {assert.equal(quota, 500, "Quota doesn't match!"); }).then( function() {return conference.numRegistrants.call();}).then( function(num) {assert.equal(num, 0, "Registrants should be zero!");return conference.organizer.call();}).then( function(organizer) {assert.equal(organizer, accounts[0], "Owner doesn't match!");done(); // to stop these tests earlier, move this up}).catch(done);}).catch(done);});});構(gòu)造函數(shù)。?Conference.new({ from: accounts[0] })通過調(diào)用合約構(gòu)造函數(shù)創(chuàng)造了一個(gè)新的Conference實(shí)例。由于不指定from時(shí)會(huì)默認(rèn)使用accounts[0],它其實(shí)可以被省略掉:
Conference.new({ from: accounts[0] }); // 和Conference.new()效果相同Promise.?代碼中的那些then和return就是Promise。它們的作用寫成一個(gè)深深的嵌套調(diào)用鏈的話會(huì)是這樣:
conference.numRegistrants.call().then(function(num) {assert.equal(num, 0, "Registrants should be zero!");conference.organizer.call().then(function(organizer) {assert.equal(organizer, accounts[0], "Owner doesn't match!");}).then(function(...))}).then(function(...))// Because this would get hairy...Promise減少嵌套,使代碼變得扁平,允許調(diào)用異步返回,并且簡(jiǎn)化了表達(dá)“成功時(shí)做這個(gè)”和“失敗時(shí)做那個(gè)”的語法。Web3.js通過回調(diào)函數(shù)實(shí)現(xiàn)異步調(diào)用,因此你不需要等到交易完成就可以繼續(xù)執(zhí)行前端代碼。Truffle借助了用Promise封裝web3.js的一個(gè)框架,叫做Pudding,這個(gè)框架本身又是基于Bluebird的,它支持Promise的高級(jí)特性。
call.?我們使用call來檢查變量的值,例如conference.quota.call().then(...,還可以通過傳參數(shù),例如call(0), 來獲取mapping在index 0處的元素。Solidity的文檔說這是一種特殊的“消息調(diào)用”因?yàn)?1.不會(huì)為礦工記錄和 2.不需要從錢包賬戶/地址發(fā)起(因此它沒有被賬戶持有者私鑰做簽名)。另一方面,交易/事務(wù)(Transaction)會(huì)被礦工記錄,必須來自于一個(gè)賬戶(也就是有簽名),會(huì)被記錄到區(qū)塊鏈上。對(duì)合約中數(shù)據(jù)做的任何修改都是交易。僅僅是檢查一個(gè)變量的值則不是。因此在讀取變量時(shí)不要忘記加上call()!否則會(huì)發(fā)生奇怪的事情。(此外如果在讀取變量是遇到問題別忘記檢查它是否是public。)call()也能用于調(diào)用不是交易的函數(shù)。如果一個(gè)函數(shù)本來是交易,但你卻用call()來調(diào)用,則不會(huì)在區(qū)塊鏈上產(chǎn)生交易。
斷言。?標(biāo)準(zhǔn)JS測(cè)試中的斷言(如果你不小心拼成了復(fù)數(shù)形式'asserts',truffle會(huì)報(bào)錯(cuò),讓你一頭霧水),assert.equal是最常用的,其他類型的斷言可以在Chai的文檔中找到。
再一次運(yùn)行truffle test確保一切工作正常。
測(cè)試合約函數(shù)調(diào)用
現(xiàn)在我們測(cè)試一下改變quote變量的函數(shù)能工作。在tests/conference.js文件的contract('Conference', function(accounts) {...};)的函數(shù)體中添加如下測(cè)試用例:
it("Should update quota", function(done) {var c = Conference.at(Conference.deployed_address);Conference.new({from: accounts[0] }).then(function(conference) {conference.quota.call().then( function(quota) { assert.equal(quota, 500, "Quota doesn't match!"); }).then( function() { return conference.changeQuota(300);}).then( function(result) { // result here is a transaction hashconsole.log(result); // if you were to print this out it’d be long hex - the transaction hashreturn conference.quota.call()}).then( function(quota) { assert.equal(quota, 300, "New quota is not correct!");done();}).catch(done);}).catch(done); });這里的新東西是調(diào)用changeQuota函數(shù)的那一行。console.log對(duì)于調(diào)試很有用,用它能在運(yùn)行truffle的終端中輸出信息。在關(guān)鍵點(diǎn)插入console.log可以查看執(zhí)行到了哪一步。記得把Solidity合約中changeQuota函數(shù)被聲明為public,否則你不能調(diào)用它:
function changeQuota(uint newquota) public { }測(cè)試交易
現(xiàn)在讓我們調(diào)用一個(gè)需要發(fā)起人發(fā)送資金的函數(shù)。
Wei.?以太幣有很多種單位(這里有個(gè)很有用的轉(zhuǎn)換器),在合約中通常用的是Wei,最小的單位。Web3.js提供了在各單位與Wei之間互相轉(zhuǎn)換的便利方法,形如web3.toWei(.05, 'ether')。JavaScript在處理很大的數(shù)字時(shí)有問題,因此web3.js使用了程序庫BigNumber,并建議在代碼各處都以Wei做單位,直到要給用戶看的時(shí)候(文檔。
賬戶余額。?Web3.js提供了許多提供方便的方法,其中另一個(gè)會(huì)在下面測(cè)試用到的是web3.eth.getBalance(some_address)。記住發(fā)送給合約的資金會(huì)由合約自己持有直到調(diào)用suicide。
在contract(Conference, function(accounts) {...};)的函數(shù)體中插入下面的測(cè)試用例。在高亮顯示的方法中,測(cè)試用例讓另一個(gè)用戶(accounts[1])以ticketPrice的價(jià)格買了一張門票。然后它檢查合約的賬戶余額增加了ticketPrice,以及購票用戶被加入了參會(huì)者列表。
這個(gè)測(cè)試中的buyTicket是一個(gè)交易函數(shù):
it("Should let you buy a ticket", function(done) {var c = Conference.at(Conference.deployed_address);Conference.new({ from: accounts[0] }).then(function(conference) {var ticketPrice = web3.toWei(.05, 'ether');var initialBalance = web3.eth.getBalance(conference.address).toNumber();conference.buyTicket({ from: accounts[1], value: ticketPrice }).then(function() {var newBalance = web3.eth.getBalance(conference.address).toNumber();var difference = newBalance - initialBalance;assert.equal(difference, ticketPrice, "Difference should be what was sent");return conference.numRegistrants.call();}).then(function(num) {assert.equal(num, 1, "there should be 1 registrant");return conference.registrantsPaid.call(accounts[1]);}).then(function(amount) {assert.equal(amount.toNumber(), ticketPrice, "Sender's paid but is not listed");done();}).catch(done);}).catch(done); });交易需要簽名。?和之前的函數(shù)調(diào)用不同,這個(gè)調(diào)用是一個(gè)會(huì)發(fā)送資金的交易,在這種情況下購票用戶(accounts[1])會(huì)用他的私鑰對(duì)buyTicket()調(diào)用做簽名。(在geth中用戶需要在發(fā)送資金之前通過輸入密碼來批準(zhǔn)這個(gè)交易或是解鎖錢包的賬戶。)
toNumber().?有時(shí)我們需要把Solidity返回的十六進(jìn)制結(jié)果轉(zhuǎn)碼。如果結(jié)果可能是個(gè)很大的數(shù)字可以用web3.toBigNumber(numberOrHexString)來處理因?yàn)镴avaScript直接對(duì)付大數(shù)要糟。
測(cè)試包含轉(zhuǎn)賬的合約
最后,為了完整性,我們確認(rèn)一下refundTicket方法能正常工作,而且只有會(huì)議組織者能調(diào)用。下面是測(cè)試用例:
it("Should issue a refund by owner only", function(done) {var c = Conference.at(Conference.deployed_address);Conference.new({ from: accounts[0] }).then(function(conference) {var ticketPrice = web3.toWei(.05, 'ether');var initialBalance = web3.eth.getBalance(conference.address).toNumber(); conference.buyTicket({ from: accounts[1], value: ticketPrice }).then(function() {var newBalance = web3.eth.getBalance(conference.address).toNumber();var difference = newBalance - initialBalance;assert.equal(difference, ticketPrice, "Difference should be what was sent"); // same as before up to here// Now try to issue refund as second user - should failreturn conference.refundTicket(accounts[1], ticketPrice, {from: accounts[1]}); }).then(function() {var balance = web3.eth.getBalance(conference.address).toNumber();assert.equal(web3.toBigNumber(balance), ticketPrice, "Balance should be unchanged");// Now try to issue refund as organizer/owner - should workreturn conference.refundTicket(accounts[1], ticketPrice, {from: accounts[0]}); }).then(function() {var postRefundBalance = web3.eth.getBalance(conference.address).toNumber();assert.equal(postRefundBalance, initialBalance, "Balance should be initial balance");done();}).catch(done);}).catch(done);});這個(gè)測(cè)試用例覆蓋的Solidity函數(shù)如下:
function refundTicket(address recipient, uint amount) public returns (bool success) {if (msg.sender != organizer) { return false; }if (registrantsPaid[recipient] == amount) { address myAddress = this;if (myAddress.balance >= amount) { recipient.send(amount);Refund(recipient, amount);registrantsPaid[recipient] = 0;numRegistrants--;return true;}}return false; }合約中發(fā)送以太幣。?address myAddress = this展示了如何獲取該會(huì)議合約實(shí)例的地址,以變接下來檢查這個(gè)地址的余額(或者直接使用this.balance)。合約通過recipient.send(amount)方法把資金發(fā)回了購票人。
交易無法返回結(jié)果給web3.js.?注意這一點(diǎn)!refundTicket函數(shù)會(huì)返回一個(gè)布爾值,但是這在測(cè)試中無法檢查。因?yàn)檫@個(gè)方法是一個(gè)交易函數(shù)(會(huì)改變合約內(nèi)數(shù)據(jù)或是發(fā)送以太幣的調(diào)用),而web3.js得到的交易運(yùn)行結(jié)果是一個(gè)交易哈希(如果打印出來是一個(gè)長(zhǎng)長(zhǎng)的十六進(jìn)制/怪怪的字符串)。既然如此為什么還要讓refundTicket返回一個(gè)值?因?yàn)樵赟olidity合約內(nèi)可以讀到這個(gè)返回值,例如當(dāng)另一個(gè)合約調(diào)用refundTicket()的時(shí)候。也就是說Solidity合約可以讀取交易運(yùn)行的返回值,而web3.js不行。另一方面,在web3.js中你可以用事件機(jī)制(Event, 下文會(huì)解釋)來監(jiān)控交易運(yùn)行,而合約不行。合約也無法通過call()來檢查交易是否修改了合約內(nèi)變量的值。
關(guān)于sendTransaction().?當(dāng)你通過web3.js調(diào)用類似buyTicket()或者refundTicket()的交易函數(shù)時(shí)(使用web3.eth.sendTransaction),交易并不會(huì)立即執(zhí)行。事實(shí)上交易會(huì)被提交到礦工網(wǎng)絡(luò)中,交易代碼直到其中一位礦工產(chǎn)生一個(gè)新區(qū)塊把交易記錄進(jìn)區(qū)塊鏈之后才執(zhí)行。因此你必須等交易進(jìn)入?yún)^(qū)塊鏈并且同步回本地節(jié)點(diǎn)之后才能驗(yàn)證交易執(zhí)行的結(jié)果。用testrpc的時(shí)候可能看上去是實(shí)時(shí)的,因?yàn)闇y(cè)試環(huán)境很快,但是正式網(wǎng)絡(luò)會(huì)比較慢。
事件/Event.?在web3.js中你應(yīng)該監(jiān)聽事件而不是返回值。我們的智能合約示例定義了這些事件:
event Deposit(address _from, uint _amount); event Refund(address _to, uint _amount);它們?cè)赽uyTicket()和refundTicket()中被觸發(fā)。觸發(fā)時(shí)你可以在testrpc的輸出中看到日志。要監(jiān)聽事件,你可以使用web.js監(jiān)聽器(listener)。在寫本文時(shí)我還不能在truffle測(cè)試中記錄事件,但是在應(yīng)用中沒問題:
Conference.new({ from: accounts[0] }).then(function(conference) {var event = conference.allEvents().watch({}, ''); // or use conference.Deposit() or .Refund()event.watch(function (error, result) {if (error) {console.log("Error: " + error);} else {console.log("Event: " + result.event);}});// ...過濾器/Filter.?監(jiān)聽所有事件可能會(huì)產(chǎn)生大量的輪詢,作為替代可以使用過濾器。它們可以更靈活的開始或是停止對(duì)事件的監(jiān)聽。更多過濾器的信息可查看Solidity文檔。
總的來說,使用事件和過濾器的組合比檢查變量消耗的Gas更少,因而在驗(yàn)證正式網(wǎng)絡(luò)的交易運(yùn)行結(jié)果時(shí)非常有用。
Gas.?(譯注:以太坊上的燃料,因?yàn)榇a的執(zhí)行必須消耗Gas。直譯為汽油比較突兀,故保留原文做專有名詞。)直到現(xiàn)在我們都沒有涉及Gas的概念,因?yàn)樵谑褂胻estrpc時(shí)通常不需要顯式的設(shè)置。當(dāng)你轉(zhuǎn)向geth和正式網(wǎng)絡(luò)時(shí)會(huì)需要。在交易函數(shù)調(diào)用中可以在{from: __, value: __, gas: __}對(duì)象內(nèi)設(shè)置Gas參數(shù)。Web3.js提供了web3.eth.gasPrice調(diào)用來獲取當(dāng)前Gas的價(jià)格,Solidity編譯器也提供了一個(gè)參數(shù)讓你可以從命令行獲取合約的Gas開銷概要:solc --gas YouContract.sol。下面是Conference.sol的結(jié)果:
為合約創(chuàng)建DApp界面
下面的段落會(huì)假設(shè)你沒有網(wǎng)頁開發(fā)經(jīng)驗(yàn)。
上面編寫的測(cè)試用例用到的都是在前端界面中也可以用的方法。你可以把前端代碼放到app/目錄中,運(yùn)行truffle build之后它們會(huì)和合約配置信息一起編譯輸出到build/目錄。在開發(fā)時(shí)可以使用truffle watch命令在app/有任何變動(dòng)時(shí)自動(dòng)編譯輸出到build/目錄。然后在瀏覽器中刷新頁面即可看到build/目錄中的最新內(nèi)容。(truffle serve可以啟動(dòng)一個(gè)基于build/目錄的網(wǎng)頁服務(wù)器。)
app/目錄中有一些樣板文件幫助你開始:
index.html會(huì)加載app.js:
因此我們只需要添加代碼到app.js就可以了。
默認(rèn)的app.js會(huì)在瀏覽器的console(控制臺(tái))中輸出一條"Hello from Truffle!"的日志。在項(xiàng)目根目錄中運(yùn)行truffle watch,然后在瀏覽器中打開build/index.html文件,再打開瀏覽器的console就可以看到。(大部分瀏覽器例如Chrome中,單擊右鍵 -> 選擇Inspect Element然后切換到Console即可。)
在app.js中,添加一個(gè)在頁面加載時(shí)會(huì)運(yùn)行的window.onload調(diào)用。下面的代碼會(huì)確認(rèn)web3.js已經(jīng)正常載入并顯示所有可用的賬戶。(注意:你的testrpc節(jié)點(diǎn)應(yīng)該保持運(yùn)行。)
window.onload = function() {var accounts = web3.eth.accounts;console.log(accounts); }看看你的瀏覽器console中看看是否打印出了一組賬戶地址。
現(xiàn)在你可以從tests/conference.js中復(fù)制一些代碼過來(去掉只和測(cè)試有關(guān)的斷言),將調(diào)用返回的結(jié)果輸出到console中以確認(rèn)代碼能工作。下面是個(gè)例子:
window.onload = function() {var accounts = web3.eth.accounts;var c = Conference.at(Conference.deployed_address);Conference.new({ from: accounts[0] }).then(function(conference) {var ticketPrice = web3.toWei(.05, 'ether');var initialBalance = web3.eth.getBalance(conference.address).toNumber(); console.log("The conference's initial balance is: " + initialBalance);conference.buyTicket({ from: accounts[1], value: ticketPrice }).then(function() {var newBalance = web3.eth.getBalance(conference.address).toNumber();console.log("After someone bought a ticket it's: " + newBalance);return conference.refundTicket(accounts[1], ticketPrice, {from: accounts[0]});}).then(function() { var balance = web3.eth.getBalance(conference.address).toNumber();console.log("After a refund it's: " + balance);});}); };上面的代碼應(yīng)該輸出如下:
(console輸出的warning信息可忽略。)
現(xiàn)在起你就可以使用你喜歡的任何前端工具,jQuery, ReactJS, Meteor, Ember, AngularJS,等等等等,在app/目錄中構(gòu)建可以與以太坊智能合約互動(dòng)的DApp界面了!接下來我們給出一個(gè)極其簡(jiǎn)單基于jQuery的界面作為示例。
這里是index.html的代碼,這里是app.js的代碼。
通過界面測(cè)試了智能合約之后我意識(shí)到最好加入檢查以保證相同的用戶不能注冊(cè)兩次。另外由于現(xiàn)在是運(yùn)行在testrpc節(jié)點(diǎn)上,速度很快,最好是切換到geth節(jié)點(diǎn)并確認(rèn)交易過程依然能及時(shí)響應(yīng)。否則的話界面上就應(yīng)該顯示提示信息并且在處理交易時(shí)禁用相關(guān)的按鈕。
嘗試geth.?如果你使用geth, 可以嘗試以下面的命令啟動(dòng) - 在我這兒(geth v1.2.3)工作的很好:
build/bin/geth --rpc --rpcaddr="0.0.0.0" --rpccorsdomain="*" --mine --unlock='0 1' --verbosity=5 --maxpeers=0 --minerthreads='4' --networkid '12345' --genesis test-genesis.json這條命令解鎖了兩個(gè)賬戶,?0和1。1. 在geth控制臺(tái)啟動(dòng)后你可能需要輸入這兩個(gè)賬戶的密碼。2. 你需要在test-genesis.json文件里面的'alloc'配置中加入你的這兩個(gè)賬戶,并且給它們充足的資金。3. 最后,在創(chuàng)建合約實(shí)例時(shí)加上gas參數(shù):
Conference.new({from: accounts[0], gas: 3141592})然后把整個(gè)truffle deploy,?truffle build流程重來一遍。
教程中的代碼。?在這篇基礎(chǔ)教程中用到的所有代碼都可以在這個(gè)代碼倉庫中找到。
自動(dòng)為合約生成界面。?SilentCicero制作了一個(gè)叫做DApp Builder的工具,可以用Solidity合約自動(dòng)生成HTML, jQuery和web.js的代碼。這種模式也正在被越來越多的正在開發(fā)中的開發(fā)者工具采用。
教程到此結(jié)束!?最后一章我們僅僅學(xué)習(xí)了一套工具集,主要是Truffle和testrpc. 要知道即使在ConsenSys內(nèi)部,不同的開發(fā)者使用的工具和框架也不盡相同。你可能會(huì)發(fā)現(xiàn)更適合你的工具,這里所說的工具可能很快也會(huì)有改進(jìn)。但是本文介紹的工作流程幫助我走上了DApp開發(fā)之路。
感謝Joseph Chow的校閱和建議,Christian Lundkvist, Daniel Novy, Jim Berry, Peter Borah和Tim Coulter幫我修改文字和debug,以及Tim Coulter, Nchinda Nchinda和Mike Goldin對(duì)DApp前端步驟圖提供的幫助。
總結(jié)
以上是生活随笔為你收集整理的教程 | 以太坊智能合约编程之菜鸟教程的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 未来社会是什么样子的?您猜对了吗?
- 下一篇: EOS大神,C++写的高发并行区块链