基础知识
Geth
Geth是由以太坊基金会提供的官方客户端软件,用Go编程语言编写的。
Solidity
Solidity是一种合约导向式语言,可被应用于各种不同的区块链平台上,其主要开发者为Gavin Wood,Christian Reitwiessner,Alex Beregszaszi,Liana Husikyan,Yoichi Hirai和其他几位早期以太坊核心贡献者。Solidity 可使程式开发人员能在区块链上(例如以太坊)编写智能合约。
Solidity的语法概念最早是由Gavin Wood在2014年提出,后期则以Christian Reitwiessner所领导的以太坊团队Solidity接手开发。该语言是针对以太坊虚拟机(EVM)所设计的四种语言之一(其他的还有Serpent,LLL,Viper(实验中)和Mutan(已弃用))。
Web3.js
javascript库,可以用来与一个节点进行交互。 由于它是一个 JavaScript 库,您可以使用它来构建基于Web的dapps。
Remix
以太坊官方推荐的智能合约开发IDE,适合新手,可以在浏览器中快速部署测试智能合约。
搭建开发环境
geth安装
Mac下安装geth
brew tap ethereum/ethereum
brew install ethereum
geth使用
geth启动一个以太坊网络节点
geth --datadir testNet --dev console 2>> test.log
启动成功界面如下图所示:
准备账户
查看账户:
> eth.accounts
我们能看到返回一个账户数组,里面有一个默认账户,如:
["0xcc4164b8535eb7a9291486a1a5c982fd15c8b75d"]
查看余额:
> eth.getBalance(eth.accounts[0])
我们默认取第一个账户的余额,容易看到这个默认分配的账号余额很多,为了更容易看到余额的变化,我们可以新创建一个账户。
创建账户:
personal.newAccount("password123456")
password123456
为新账户的密码,这时我们输出账户数组就能看到两个账户了,输出新账户的余额eth.getBalance(eth.accounts[1])
,我们能看到值为0,用新账户部署合约是需要有余额的,所以我们需要转一个以太币到新账户里面。
给新账户转币:
eth.sendTransaction({from: '0xcc4164b8535eb7a9291486a1a5c982fd15c8b75d', to: '0x0a9961a0cde47c7e806a2874dc065cb33f6201c5', value: web3.toWei(1, "ether")})
然后你就能在你的test.log
日志里看到挖矿记录了。再次查看余额eth.getBalance(eth.accounts[1])
就能看到账户里有一个以太币了。
解锁账户:
在部署合约前需要先解锁账户,输入你的账号还有之前设置的密码,使用以下命令:
personal.unlockAccount(eth.accounts[1],"password123456");
编写合约代码
部署合约
以下是一个简单的智能合约代码:
pragma solidity 0.4.24;
contract hello {
function mutiply(uint a) returns (uint result) {
return a*3;
}
}
这段代码就是传入一个数字,然后每次调用mutiply()的时候返回这个数字乘以3的结果。
把这段代码复制进入remix,这个就是一个浏览器ide。这个浏览器ide也可以安装在本地,这里面有教程教你具体怎么安装:github地址。点击Details即可部署智能合约代码,然后再弹框里找到以下这段代码,直接复制,修改里面的web.eth.accounts[0]
为web.eth.accounts[1]
。
然后在命令行里面输入以上代码:
> var helloContract = web3.eth.contract([{"constant":true,"inputs":[{"name":"a","type":"uint256"}],"name":"mutiply","outputs":[{"name":"result","type":"uint256"}],"payable":false,"stateMutability":"nonpayable","type":"function"}]);
undefined
> var hello = helloContract.new(
{
from: web3.eth.accounts[1],
data: '0x608060405234801561001057600080fd5b5060bb8061001f6000396000f300608060405260043610603f576000357c0100000000000000000000000000000000000000000000000000000000900463ffffffff168063f70d290d146044575b600080fd5b348015604f57600080fd5b50606c600480360381019080803590602001909291905050506082565b6040518082815260200191505060405180910390f35b60006003820290509190505600a165627a7a7230582002abd0500936c002b085e7f017e1aca58ca5294e5e4b27f0733e430604076c3f0029',
gas: '4700000'
}, function (e, contract){
console.log(e, contract);
if (typeof contract.address !== 'undefined') {
console.log('Contract mined! address: ' + contract.address + ' transactionHash: ' + contract.transactionHash);
}
})
执行代码后输出以下结果表示合约部署成功,address就是合约地址:
> null [object Object]
Contract mined! address: 0xa29b84d8d302820f6ca1ebbf2f159ba12cf82b02 transactionHash: 0x15dd29aa336c8c4bf5868422c202f55153990a04426d5130e454957a87f56b07
调用合约
部署完合约后,我们就可以开始调用合约了,引号里面的值就是合约的地址,这里定义了一个Demo变量,然后通过Demo调用方法multiply()
> Demo=eth.contract(hello.abi).at("0xa29b84d8d302820f6ca1ebbf2f159ba12cf82b02")
{
abi: [{
constant: true,
inputs: [{...}],
name: "mutiply",
outputs: [{...}],
payable: false,
stateMutability: "nonpayable",
type: "function"
}],
address: "0xa29b84d8d302820f6ca1ebbf2f159ba12cf82b02",
transactionHash: null,
allEvents: function(),
mutiply: function()
}
> Demo.mutiply(3)
9