创世区块准备
关于geth的安装并没有遇到什么问题,按照网上教程通过命令行安装。
通过查阅网络资源和官方文档,了解到要运行自己的私有链,就需要定义自己的创世区块。这个创世区块的信息需要提前编写在josn格式的配置文件中,这部分内容我直接参考官网,下面是我的genesis.json配置文件内容:
{
"config": {
"chainId": 10, /*官网上这个参数为0*/
"homesteadBlock": 0,
"eip155Block": 0,
"eip158Block": 0
},
"alloc" : {},
"coinbase" : "0x0000000000000000000000000000000000000000",
"difficulty" : "0x20000",
"extraData" : "",
"gasLimit" : "0x2fefd8",
"nonce" : "0x0000000000000042",
"mixhash" : "0x0000000000000000000000000000000000000000000000000000000000000000",
"parentHash" : "0x0000000000000000000000000000000000000000000000000000000000000000",
"timestamp" : "0x00"
}
参数解释
chainId: 这个参数指定链独立的区块链网络ID,在连接到其他节点时会用得到,以太坊公网的网络ID是1,为不与公有链冲突,运行私有链节点时需要自己指定网络ID
HomesteadBlock: 当设置为0表示使用Homestead发布该链
nonce: 一个64位随机数,用于挖矿
mixhash: 与nonce配合用于挖矿,由上一个区块的一部分生成的hash
difficulty: 设置设置当前区块的难度,值越大挖矿就越难
alloc: 用来预置账号以及账号的以太币数量
coinbase: 矿工账号
timestamp: 设置创世块的时间戳
parentHash: 上一个区块的hash,创世块就为0
extraData: 附加信息,自己可以填写任意信息
gasLimit: 该值设置对GAS的消耗总量限制,用来限制区块能包含的交易信息总和
完成上述操作后,将genesis.json存放到合适的目录下,进入该目录后,执行下面的命令后,创建创世区块:
geth --datadir "./"init genesis.json
运行成功后会生成两个文件夹,geth用来存放链上的区块数据,keystone用来存放用户信息,目录结构如下:
..
├── geth
│ │ └── chaindata
│ │ ├── 000002.ldb
│ │ ├── 000003.log
│ │ ├── CURRENT
│ │ ├── LOCK
│ │ ├── LOG
│ │ └── MANIFEST-000004
│ └── keystore
此时其实已经初始化成功一条自己的私有链,之后需要将其启动,进行操作,命令如下:
geth --networkid 1108 --nodiscover --datadir “./“ --rpc --rpcapi "net,eth,web3,personal" --rpcaddr "0.0.0.0" --rpccorsdomain "*" --rpcport 8545 console 2>>detial1.log
参数解释
--networked :指定私有链的网络ID为1108
--nodiscover :使自己的节点不会被其他人发现
--datadir :私有链存放路径
--rpc :允许RPC操作自己的节点
--rpcapi :确定允许通过RPC访问的命令
--rpcaddr :rpc接口的地址,此处设置的参数是为了保证remix可以链接私有链
--rpcport :rpc接口的端口号
console :指定启动成功后进入命令行界面
2>> :将日志重定向到指定文件中
启动成功后会提示当前Geth版本号,矿工账号,区块数,以及存放目录等
difficulty :当前块的难度
extraData :当前块的extraData字段值
gasLimit :当前区块允许使用的最大gas值
gasUsed :当前区块累计使用的总gas值
hash :区块的哈希串,当这个区块处于pending时将返回null
miner :获得挖矿奖励的矿工
mixHash :
nonce :POW生成的哈希,当此区块处于pending时将返回null
parentHash :父区块的哈希值
sha3Uncles :叔区块的哈希值
size :当前区块的字节大小
stateRoot :区块的最终状态前缀树的根
timestamp :区块打包时的unix时间戳
transactions :交易对象,是一个数组
transactionsRoot :交易对象的根哈希
uncles :叔哈希的数组
合约部署
contract Token {
address issuer;
mapping (address => uint) balances;
event Issue(address account, uint amount);
event Transfer(address from, address to, uint amount);
function Token() {
issuer = msg.sender;
}
function issue(address account, uint amount) {
if (msg.sender != issuer) throw;
balances[account] += amount;
}
function transfer(address to, uint amount) {
if (balances[msg.sender] < amount) throw;
balances[msg.sender] -= amount;
balances[to] += amount;
Transfer(msg.sender, to, amount);
}
function getBalance(address account) constant returns (uint) {
return balances[account];
}
}
这段代码实现了一个简单的Token功能,issue函数可以向合约账户充值以太;transfer函数可以向其他账号发送token;而getBalance可以获取某个账号的token余额,相当于通过eth.getBalance(eth.account[])来获取某个账户的信息。
将这段简单的合约通过web端的remix编译成功后,在私有链进行部署,在Enveironment选择Web3 Provider,remix就能发现私有链,此时编译器中的Account显示的会是自己在私有链中创建的账户以及账户余额。
随后点击Deploy对合约进行部署,此时需要在终端通过miner.start(1)执行一段时间后停止,之后会在remix看到输出日志:
将pending(表示已提交并未处理的交易)通过挖矿处理后,会输出交易的详情,此时表明合约已经成功部署到私链上;下面两次pending分别是向特定账户充值和进行转账,将pending处理后,输出交易详情
对一次交易的字段解释
blockHash :区块哈希值
blockNumber :区块号
from :交易发送者的地址
gas :表示这个交易允许消耗的最大gas数量
gasprice :表示发送者愿意支付给矿工的gas价格
hash :由交易信息生成的哈希,是交易ID
input :存在的数据字段,如果存在,则是表明该交易是一个创建或者调用智能合约交易
to :交易接收者的地址
value :发送者要转移给接收者的以太币数量
nonce :用来区别同一用户发出的不同交易的标记
r,s,v :交易签名的三个部分,由发送者的私钥对交易hash进行签名生成