以太坊私链搭建及智能合约部署

创世区块准备

关于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 Providerremix就能发现私有链,此时编译器中的Account显示的会是自己在私有链中创建的账户以及账户余额。

image

随后点击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进行签名生成

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

友情链接更多精彩内容