智能合約開發環境搭建及Hello World合約

本文首發於深刻淺出區塊鏈社區
原文連接:智能合約開發環境搭建及Hello World合約原文已更新,請讀者前往原文閱讀html

若是你對於以太坊智能合約開發尚未概念(本文會假設你已經知道這些概念),建議先閱讀入門篇
就先學習任何編程語言同樣,入門的第一個程序都是Hello World。今天咱們來一步一步從搭建以太坊智能合約開發環境開始,講解智能合約的Hello World如何編寫。git

開發環境搭建

Solidity安裝

強烈建議新手使用Browser-Solidity來進行開發。
Browser-Solidity是一個基於瀏覽器的Solidity,就能夠不用安裝Solidity,本文的Hello World教程也將基於Browser-Solidity來進行。github

若是你想本身安裝請參考Solidity安裝指引web

geth 安裝

Mac下安裝命令以下:其餘平臺參考:geth官方安裝指引編程

brew tap ethereum/ethereum
brew install ethereum
brew 是 Mac 下的包管理工具,和Ubuntu裏的apt-get相似

安裝完之後,就是把geth控制檯啓動。數組

啓動環境

在入門篇講過,geth是一個以太坊客戶端,如今利用geth啓動一個以太坊(開發者)網絡節點。瀏覽器

geth --datadir testNet --dev console 2>> test.log

執行命名後,會進入geth控制檯,這時光標停在一個向右的箭頭處,像這樣:
網絡

命令參數說明(更多命令詳解可閱讀Geth命令用法-參數詳解篇):
--dev 啓用開發者網絡(模式),開發者網絡會使用POA共識,默認預分配一個開發者帳戶而且會自動開啓挖礦。
--datadir 後面的參數是區塊數據及祕鑰存放目錄。
第一次輸入命令後,它會放在當前目錄下新建一個testNet目錄來存放數據。
console 進入控制檯
2>> test.log 表示把控制檯日誌輸出到test.log文件編程語言

爲了更好的理解,建議新開一個命令行終端,實時顯示日誌:編輯器

tail -f test.log

準備帳戶

部署智能合約須要一個外部帳戶,咱們先來看看分配的開發者帳戶,在控制檯使用如下命令查看帳戶:

> eth.accounts

回車後,返回一個帳戶數組,裏面有一個默認帳戶,如:

也可使用personal.listAccounts查看帳戶,

再來看一下帳戶裏的餘額,使用一下命令:

> eth.getBalance(eth.accounts[0])

eth.accounts[0]表示帳戶列表第一個帳戶
回車後,能夠看到大量的餘額,如:
1.15792089237316195423570985008687907853269... e+77

開發者帳戶因餘額太多,若是用這個帳戶來部署合約時會沒法看到餘額變化,爲了更好的體驗完整的過程,這裏選擇建立一個新的帳戶。

建立帳戶

使用如下命令建立帳戶:

> personal.newAccount("TinyXiong")

TinyXiong爲新帳戶的密碼,回車後,返回一個新帳戶。

這時咱們查看帳戶列表:

> eth.accounts

能夠看到帳戶數組你包含兩個帳戶,新帳戶在第二個(索引爲1)位置。

如今看看帳戶的餘額:

> eth.getBalance(eth.accounts[1])
0

回車後,返回的是0,新帳戶是0。結果如:

給新帳戶轉帳

咱們知道沒有餘額的帳戶是無法部署合約的,那咱們就從默認帳戶轉1以太幣給新帳戶,使用如下命令(請使用你本身eth.accounts對應輸出的帳戶):

eth.sendTransaction({from: '0xb0ebe17ef0e96b5c525709c0a1ede347c66bd391', to: '0xf280facfd60d61f6fd3f88c9dee4fb90d0e11dfc', value: web3.toWei(1, "ether")})
在打開的 tail -f test.log日誌終端裏,能夠同時看到挖礦記錄
再次查看新帳戶餘額,能夠新帳戶有1個以太幣

解鎖帳戶

在部署合約前須要先解鎖帳戶(就像銀行轉帳要輸入密碼同樣),使用如下命令:

personal.unlockAccount(eth.accounts[1],"TinyXiong");

"TinyXiong" 是以前建立帳戶時的密碼
解鎖成功後,帳戶就準備完畢啦,接下來就是編寫合約代碼。

編寫合約代碼

如今咱們來開始編寫第一個智能合約代碼,solidity代碼以下:

pragma solidity ^0.4.18;
contract hello {
    string greeting;
    
    function hello(string _greeting) public {
        greeting = _greeting;
    }

    function say() constant public returns (string) {
        return greeting;
    }
}

簡單解釋下,咱們定義了一個名爲hello的合約,在合約初始化時保存了一個字符串(咱們會傳入hello world),每次調用say返回字符串。
把這段代碼寫(拷貝)到Browser-Solidity,若是沒有錯誤,點擊Details獲取部署代碼,如:

在彈出的對話框中找到WEB3DEPLOY部分,點拷貝,粘貼到編輯器後,修改初始化字符串爲hello world。

solidity在博文寫做時(2017/11/24),版本爲0.4.18,solidity發展很是快,solidity版本之間有可能不能兼容,這是你能夠在Browser-Solidity的Settings裏選擇對應的編譯器版本。
Browser-Solidity也不停的更新中,截圖可能和你看到的界面不同。

部署合約

Browser-Solidity生成的代碼,拷貝到編輯器裏修改後的代碼以下:

var _greeting = "Hello World" ;
var helloContract = 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"}]);
var hello = helloContract.new(
   _greeting,
   {
     from: web3.eth.accounts[1], 
     data: '0x6060604052341561000f57600080fd5b6040516102b83803806102b8833981016040528080518201919050508060009080519060200190610041929190610048565b50506100ed565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282601f1061008957805160ff19168380011785556100b7565b828001600101855582156100b7579182015b828111156100b657825182559160200191906001019061009b565b5b5090506100c491906100c8565b5090565b6100ea91905b808211156100e65760008160009055506001016100ce565b5090565b90565b6101bc806100fc6000396000f300606060405260043610610041576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063954ab4b214610046575b600080fd5b341561005157600080fd5b6100596100d4565b6040518080602001828103825283818151815260200191508051906020019080838360005b8381101561009957808201518184015260208101905061007e565b50505050905090810190601f1680156100c65780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6100dc61017c565b60008054600181600116156101000203166002900480601f0160208091040260200160405190810160405280929190818152602001828054600181600116156101000203166002900480156101725780601f1061014757610100808354040283529160200191610172565b820191906000526020600020905b81548152906001019060200180831161015557829003601f168201915b5050505050905090565b6020604051908101604052806000815250905600a165627a7a723058204a5577bb3ad30e02f7a3bdd90eedcc682700d67fc8ed6604d38bb739c0655df90029', 
     gas: '4700000'
   }, function (e, contract){
    console.log(e, contract);
    if (typeof contract.address !== 'undefined') {
         console.log('Contract mined! address: ' + contract.address + ' transactionHash: ' + contract.transactionHash);
    }
 });

第1行:修改字符串爲Hello World
第2行:修改合約變量名
第3行:修改合約實例變量名,以後能夠直接用實例調用函數。
第6行:修改部署帳戶爲新帳戶索引,即便用新帳戶來部署合約。
第8行:準備付的gas費用,IDE已經幫咱們預估好了。
第9行:設置部署回調函數。

拷貝回geth控制檯裏,回車後,看到輸出如:

Contract mined! address: 0x79544078dcd9d560ec3f6eff0af42a9fc84c7d19 transactionHash: 0xe2caab22102e93434888a0b8013a7ae7e804b132e4a8bfd2318356f6cf0480b3

說明合約已經部署成功。

在打開的 tail -f test.log日誌終端裏,能夠同時看到挖礦記錄

如今咱們查看下新帳戶的餘額:

> eth.getBalance(eth.accounts[1])

是否是比以前轉帳的餘額少呀!

運行合約

> hello.say()
"Hello World"

輸出Hello World,咱們第一個合約Hello World,成功運行了。

運行截圖以下:

本文會隨geth,solidity語言版本升級保持更新,查看本文原始連接:https://learnblockchain.cn/2017/11/24/init-env/

第一個合約的意義更重要的是體驗智能合約開發流程,對於初學者一些能夠選擇先放棄一些細節,開發流程打通以後,能夠加強信心進行下一步的學習。
有問題就加入深刻淺出區塊鏈羣一塊兒來交流吧。

深刻淺出區塊鏈 - 系統學習區塊鏈,打造最好的區塊鏈技術博客。

☛ 個人知識星球爲各位解答區塊鏈技術問題,歡迎加入討論。

☛ 關注公衆號「深刻淺出區塊鏈技術」第一時間獲取區塊鏈技術信息。

相關文章
相關標籤/搜索