本文首发于深刻浅出区块链社区
原文连接:以太坊创世区块与链配置载入分析,原文已更新,请读者前往原文阅读。数据库
创世区块做为第零个区块,其余区块直接或间接引用到创世区块。所以节点启动之初必须载入正确的创世区块信息,且不得任意修改。json
以太坊容许经过创世配置文件来初始化创世区块,也可以使用选择使用内置的多个网络环境的创世配置。默认使用以太坊主网创世配置。segmentfault
若是你须要搭建以太坊私有链,那么了解创世配置是必须的,不然你大可不关心创世配置。下面是一份 JSON 格式的创世配置示例:bash
{ "config": { "chainId": 1, "homesteadBlock": 1150000, "daoForkBlock": 1920000, "daoForkSupport": true, "eip150Block": 2463000, "eip150Hash": "0x2086799aeebeae135c246c65021c82b4e15a2c451340993aacfd2751886514f0", "eip155Block": 2675000, "eip158Block": 2675000, "byzantiumBlock": 4370000, "constantinopleBlock": 7280000, "petersburgBlock": 7280000, "ethash": {} }, "nonce": "0x42", "timestamp": "0x0", "extraData": "0x11bbe8db4e347b4e8c937c1c8370e4b5ed33adb3db69cbdb7a38e1e50b1b82fa", "gasLimit": "0x1388", "difficulty": "0x400000000", "mixHash": "0x0000000000000000000000000000000000000000000000000000000000000000", "coinbase": "0x0000000000000000000000000000000000000000", "number": "0x0", "gasUsed": "0x0", "parentHash": "0x0000000000000000000000000000000000000000000000000000000000000000", "alloc": { "000d836201318ec6899a67540690382780743280": { "balance": "0xad78ebc5ac6200000" }, "001762430ea9c3a26e5749afdb70da5f78ddbb8c": { "balance": "0xad78ebc5ac6200000" } } }
根据配置用途可分为三大类:网络
config
项是定义链配置,会影响共识协议,虽然链配置对创世影响不大,但新区块的出块规则均依赖链配置。svg
创世区块头信息配置学习
nonce
:随机数,对应创世区块 Nonce
字段。timestamp
:UTC时间戳,对应创世区块 Time
字段。extraData
:额外数据,对应创世区块 Extra
字段。gasLimit
:必填,燃料上限,对应创世区块 GasLimit
字段。difficulty
:必填,难度系数,对应创世区块 Difficulty
字段。搭建私有链时,须要根据状况选择合适的难度值,以便调整出块。minHash
:一个哈希值,对应创世区块的MixDigest
字段。和 nonce 值一块儿证实在区块上已经进行了足够的计算。coinbase
:一个地址,对应创世区块的Coinbase
字段。alloc
项是创世中初始帐户资产配置。在生成创世区块时,将此数据集中的帐户资产写入区块中,至关于预挖矿。这对开发测试和私有链很是好用,不须要挖矿就能够直接为任意多个帐户分配资产。区块链
若是你计划部署以太坊私有网络或者一个独立的测试环境,那么须要自定义创世,并初始化它。为了统一沟通,推荐先在用户根目录建立一个文件夹 deepeth
,以作为《以太坊设计与实现》电子书学习工做目录。测试
mkdir $HOME/deepeth && cd $HOME/deepeth
再准备两个以太坊帐户,以便在创世时存入资产。ui
geth --datadir $HOME/deepeth account new
由于是学习使用,推荐使用统一密码 foobar
,执行两次命令,建立好两个帐户。这里使用 --datadir
参数指定以太坊运行时数据存放目录,是让你们将数据统一存放在一个本课程学习文件夹中。
再将下面配置内容保存到 $HOME/deepeth/genesis.json
文件,其中 alloc
项替换成刚刚建立的两个以太坊帐户地址。
{ "config": { "chainId": 8888, "homesteadBlock": 0, "daoForkBlock": 0, "daoForkSupport": true, "eip150Block": 0, "eip155Block": 0, "eip158Block": 0, "byzantiumBlock": 0, "constantinopleBlock": 0, "petersburgBlock": 0, "ethash": {} }, "nonce": "0x42", "timestamp": "0x0", "extraData": "0x11bbe8db4e347b4e8c937c1c8370e4b5ed33adb3db69cbdb7a38e1e50b1b82fa", "gasLimit": "0x1388", "difficulty": "0x1", "alloc": { "093f59f1d91017d30d8c2caa78feb5beb0d2cfaf": { "balance": "0xffffffffffffffff" }, "ddf7202cbe0aaed1c2d5c4ef05e386501a054406": { "balance": "0xffffffffffffffff" } } }
而后,执行 geth 子命令 init 初始化创世区块。
geth --datadir $HOME/deepeth init genesis.json
执行成功后,即可启动该私有链:
geth --maxpeers 0 --datadir $HOME/deepeth console
执行以下命令,能够查看到前面建立的两个帐户,均已有资产:
eth.getBalance(eth.accounts[0]) // 18446744073709551615 eth.getBalance(eth.accounts[1]) // 18446744073709551615
至此,咱们已完成创世定制版。
上面我已完成自定义创世,但以太坊做为去中心平台,须要许多节点一块儿参与。仅仅为了测试,多个节点来搭建私有链比较麻烦。若是但愿和别人一块儿联调,或者须要在测试网络中测试DAPP时,该怎么办呢?那么,可以使用以太坊测试网络。以太坊公开的测试网络有 5 个,目前仍在运行的有 4 个,具体见下表格。
测试网 | 共识机制 | 出块间隔 | 提供方 | 上线时间 | 备注 | 状态 |
---|---|---|---|---|---|---|
Morden | PoW | 以太坊官方 | 2015.7 | 因难度炸弹被迫退役 | stopped | |
Ropsten | PoW | 30秒 | 以太坊官方 | 2016.11 | 接替Morden | running |
Kovan | PoA | 4秒 | 以太坊钱包Parity开发团队 | 2017.3 | 不支持geth | running |
Rinkeby | PoA | 15秒 | 以太坊官方 | 2017.4 | 最经常使用,只支持geth | running |
Sokol | PoA | 5秒 | 以太坊官方POA.network团队 | 2017.12 | 不支持geth | running |
Görli | PoA | 15秒 | 以太坊柏林社区 | 2018.9 | 首个以太坊2.0实验场 | running |
支持 geth 的3个测试网络的创世配置已内置在以太坊代码中,具体见 core/genesis.go
文件:
// DefaultTestnetGenesisBlock returns the Ropsten network genesis block. func DefaultTestnetGenesisBlock() *Genesis{} // DefaultRinkebyGenesisBlock returns the Rinkeby network genesis block. func DefaultRinkebyGenesisBlock() *Genesis // DefaultGoerliGenesisBlock returns the Görli network genesis block. func DefaultGoerliGenesisBlock() *Genesis{}
固然不会缺以太坊主网创世配置,也是 geth 运行的默认配置。
// DefaultGenesisBlock returns the Ethereum main net genesis block. func DefaultGenesisBlock() *Genesis{}
若是你不想自定义创世配置文件用于开发测试,那么以太坊也提供一份专用于本地开发的配置。
// DeveloperGenesisBlock returns the 'geth --dev' genesis block. Note, this must // be seeded with the func DeveloperGenesisBlock(period uint64, faucet common.Address) *Genesis
运行 geth --dev console
可临时运行使用。但若是须要长期使用此模式,则须要指定 datadir
。
geth --dev --datadir $HOME/deepeth/dev console
首次运行 dev 模式会自动建立一个空密码的帐户,并开启挖矿。当有新交易时,将马上打包出块。
在运行 geth 时需根据配置文件加载创世配置以及创世区块,并校验其合法性。若是配置信息随意变动,易引发共识校验不经过等问题。只有在加载并检查经过时,才能继续运行程序。
<img src="https://img.learnblockchain.c...; width="400px" alt="创世加载流程">
上图是一个简要流程,下面分别讲解“加载创世配置”和“安装创世区块”两个子流程。
应使用哪一种创世配置,由用户在启动 geth 时决定。下图是创世配置选择流程图:
经过 geth 命令参数可选择不一样网络配置,能够经过 networkid
选择,也可以使用网络名称启用。
使用 networkid:
不一样网络使用不一样ID标识。
直接使用网络名称:
geth 启动时根据不一样参数选择加载不一样网络配置,并对应不一样网络环境。若是不作任何选择,虽然在此不会作出选择,但在后面流程中会默认使用主网配置。
上面已初步选择创世配置,而这一步则根据配置加载或者初始化创世单元。下图是处理流程:
首先,须要从数据库中根据区块高度 0 读取创世区块哈希。若是不存在则说明本地属于第一次启动,直接使用运行时创世配置来构建创世区块。属于首次,还须要存储创世区块和链配置。
若是存在,则须要使用运行时创世配置构建创世区块并和本次已存储的创世区块哈希进行对比。一旦不一致,则返回错误,不得继续。
随后,还须要检查链配置。先从数据库获取链配置,若是不存在,则无需校验直接使用运行时链配置。不然,须要检查运行时链配置是否正确,只有正确时才能替换更新。但有一个例外:主网配置不得随意更改,由代码控制而非人为指定。
总的来讲,以太坊默认使用主网配置,只有在首次运行时才建立和存储创世区块,其余时候仅仅用于校验。而链配置除主网外则在规则下可随时变动。
上面咱们已知晓整体流程,这里再细说下以太坊是如何根据创世配置生成创世区块。核心代码位于 core/genesis.go:229
。
func (g *Genesis) ToBlock(db ethdb.Database) *types.Block{ if db == nil { db = rawdb.NewMemoryDatabase() } statedb, _ := state.New(common.Hash{}, state.NewDatabase(db))//❶ for addr, account := range g.Alloc { //❷ statedb.AddBalance(addr, account.Balance) statedb.SetCode(addr, account.Code) statedb.SetNonce(addr, account.Nonce) for key, value := range account.Storage { statedb.SetState(addr, key, value) } } root := statedb.IntermediateRoot(false)//❸ head := &types.Header{//❹ Number: new(big.Int).SetUint64(g.Number), Nonce: types.EncodeNonce(g.Nonce), Time: g.Timestamp, ParentHash: g.ParentHash, Extra: g.ExtraData, GasLimit: g.GasLimit, GasUsed: g.GasUsed, Difficulty: g.Difficulty, MixDigest: g.Mixhash, Coinbase: g.Coinbase, Root: root, } //❺ if g.GasLimit == 0 { head.GasLimit = params.GenesisGasLimit } if g.Difficulty == nil { head.Difficulty = params.GenesisDifficulty } statedb.Commit(false)//❻ statedb.Database().TrieDB().Commit(root, true)//❼ return types.NewBlock(head, nil, nil, nil)//❽ }
上面代码是根据创世配置生成创世区块的代码逻辑,细节以下:
state
(后续文章会详细讲解 state
对象)。❷ 遍历配置中 Alloc
项帐户集合数据,直接写入 state 中。
这里不单能够设置 balance
,还能够设置 code
、nonce
以及任意多个 storage
数据。
意味着创世时即可以直接部署智能合约。例以下面配置则在创世时部署了一个名为093f59f1d91017d30d8c2caa78feb5beb0d2cfaf
的智能合约。
"alloc": { "093f59f1d91017d30d8c2caa78feb5beb0d2cfaf": { "balance": "0xffffffffffffffff", "nonce": "0x3", "code":"0x606060", "storage":{ "11bbe8db4e347b4e8c937c1c8370e4b5ed33adb3db69cbdb7a38e1e50b1b82fa":"1234ff" } } }
StateRoot
。Root
字段中。GasLimit
和 Difficulty
直接影响到下一个区块出块处理。这是多余的一步,由于提交到数据库是由外部进行,这里只须要负责生成区块。
深刻浅出区块链 - 系统学习区块链,学区块链都在这里,打造最好的区块链技术博客。