先定義塊和鏈的基礎(chǔ)結(jié)構(gòu)磅甩,目前不包含交易、賬戶等信息彩倚。
*block
block.go
包含的功能:
1. block的數(shù)據(jù)結(jié)構(gòu)(包括header,和body的數(shù)據(jù)結(jié)構(gòu))
- block結(jié)構(gòu)包括:Header,Body
- Header結(jié)構(gòu)包括:父hash(ParentHash)筹我、產(chǎn)生塊的節(jié)點(diǎn)的地址(Coinbase)、區(qū)塊產(chǎn)生的時(shí)間戳(Timestamp)帆离、區(qū)塊hash(Blockhash)蔬蕊、區(qū)塊號(hào)(Number)、額外信息(Extradata)
- Body結(jié)構(gòu)包括:預(yù)留字段哥谷。第一版還沒(méi)有包含交易等信息進(jìn)去岸夯。
2. 計(jì)算block的hash方法-SetHash
- 組裝的參數(shù):ParentHash,Timestamp们妥,Number猜扮,Extradata
- 將block中的已有字段的string類(lèi)型數(shù)據(jù)拼接起來(lái),轉(zhuǎn)為byte监婶,通過(guò)keccak256算法
crypto.Keccak256Hash
得到Hash
3. 創(chuàng)建新區(qū)塊函數(shù)-NewBlock
- 參數(shù):傳入header和body
- 組裝區(qū)塊
- 計(jì)算當(dāng)前塊生成的時(shí)間戳
- 計(jì)算塊的hash旅赢,返回block
4. 創(chuàng)建genesisblock函數(shù)-NewGenesisBlock
- 第一版先定義一個(gè)塊為創(chuàng)世塊。內(nèi)容先在方法里設(shè)置好
- 返回block
1. block的數(shù)據(jù)結(jié)構(gòu)(包括header,和body的數(shù)據(jù)結(jié)構(gòu))
type Header struct {
ParentHash string //父hash
Coinbase []byte //miner
Timestamp int64 //區(qū)塊產(chǎn)生的時(shí)間戳
Blockhash string //區(qū)塊hash
Number int //區(qū)塊號(hào)
Extradata []byte //額外信息
//todo:difficulty\gaslimit\gasused\nonce\totaldifficulty
//todo:uncle\stateroot\Txhash\receipthash\bloom\
}
//body先預(yù)留一個(gè)字段
type Body struct {
body []byte
}
type Block struct {
header *Header
// transactions Transactions
body *Body
}
2. 計(jì)算block的hash方法-SetHash
將block中的已有字段拼接起來(lái)惑惶,轉(zhuǎn)為byte煮盼,通過(guò)keccak256算法得到Hash值
//計(jì)算Blockhash
//將block中的已有字段拼接起來(lái),轉(zhuǎn)為byte带污,通過(guò)keccak256算法得到Hash值
//組裝的參數(shù)為每個(gè)參數(shù)的string類(lèi)型:ParentHash僵控,Timestamp,Number刮刑,Extradata
func (block *Block) SetHash() string {
header := block.header
record := string(header.Extradata) + strconv.FormatInt(header.Timestamp, 10) + strconv.Itoa(header.Number) + header.ParentHash
recordbyte := []byte(record)
h := crypto.Keccak256Hash(recordbyte).String() //轉(zhuǎn)為commonhash
return h
}
3. 創(chuàng)建新區(qū)塊函數(shù)-NewBlock
//生成新區(qū)塊
//TODO:body中的tx等作為參數(shù)傳入
func NewBlock(header *Header, body *Body) *Block {
b := &Block{header: CopyHeader(header), body: body}
b.header.Timestamp = time.Now().Unix()
b.header.Blockhash = b.SetHash()
return b
}
4. 創(chuàng)建genesisblock函數(shù)-NewGenesisBlock
//genesisblock
//todo喉祭,通過(guò)文件形式傳入genesisblock
func NewGenesisBlock() *Block {
header := &Header{}
body := &Body{}
b := &Block{header: header, body: body}
b.header.Extradata = []byte("Genesis Block")
b.header.Coinbase = nil
b.header.Number = 0
b.header.ParentHash = nil
b.header.Timestamp = 1535706356
b.header.Blockhash = b.SetHash()
Logger.Infoln("block", b)
return b
}
5. 處理header函數(shù)
//處理header,防止修改頭變量出現(xiàn)其他影響
func CopyHeader(header *Header) *Header {
copyhead := *header
if len(copyhead.Extradata) > 0 {
copyhead.Extradata = make([]byte, len(header.Extradata))
copy(copyhead.Extradata, header.Extradata)
}
return ©head
}
*blockchain
定義完區(qū)塊的一些基礎(chǔ)結(jié)構(gòu)养渴,可以開(kāi)始定義基礎(chǔ)鏈中的一些信息
blockchain.go
包含的功能:
1. blockchain的數(shù)據(jù)結(jié)構(gòu)
- 是一個(gè)block數(shù)組
2. 初始化一條新區(qū)塊鏈函數(shù)-NewBlockchain
- 定義BlockChain類(lèi)型的鏈對(duì)象
- 將創(chuàng)世塊作為鏈的第一個(gè)數(shù)組對(duì)象加進(jìn)去
3. 向鏈中添加區(qū)塊方法-AddBlock
- block對(duì)象
- 當(dāng)前區(qū)塊的ParentHash,將區(qū)塊鏈中的上一個(gè)塊的Blockhash賦值給當(dāng)前區(qū)塊的ParentHash
- 當(dāng)前區(qū)塊的塊Number泛烙,在區(qū)塊鏈中的上一個(gè)區(qū)塊的Number基礎(chǔ)上+1
- 返回block
1. blockchain的數(shù)據(jù)結(jié)構(gòu)
是一個(gè)block數(shù)組
//blockchain結(jié)構(gòu)
//是一個(gè)block數(shù)組
type BlockChain struct {
blocks []*Block
}
2. 初始化一條新區(qū)塊鏈函數(shù)-NewBlockchain
//啟動(dòng)一條區(qū)塊鏈
func NewBlockchain() *BlockChain {
return &BlockChain{[]*Block{NewGenesisBlock()}}
}
3. 向鏈中添加區(qū)塊方法-AddBlock
向鏈中添加區(qū)塊
//向區(qū)塊鏈中添加區(qū)塊
//
func (bc *BlockChain) AddBlock(header *Header, body *Body) {
prevBlock := bc.blocks[len(bc.blocks)-1]
header.ParentHash = prevBlock.header.Blockhash
header.Number = prevBlock.header.Number + 1
newBlock := NewBlock(header, body)
bc.blocks = append(bc.blocks, newBlock)
}
*main函數(shù)
main.go
主要實(shí)現(xiàn)功能:
- 添加兩個(gè)區(qū)塊
- 打印區(qū)塊鏈中的區(qū)塊信息
- 注: 這里的log用的是自己內(nèi)部使用的一個(gè)包理卑。可以用別的log包替換蔽氨,不影響藐唠,只是為了規(guī)范輸出的日志內(nèi)容。
package main
import (
mylog "mylog2"
"time"
)
var Logger *mylog.SimpleLogger
//注釋1
func main() {
Logger = mylog.NewSimpleLogger("XChain-Go")
Logger.Infoln("版本信息GitVersion:", GitVersion, ",BuildDate:", BuildDate)
//初始化一條鏈
bc := NewBlockchain()
//構(gòu)造兩個(gè)區(qū)塊
header1 := &Header{}
header2 := &Header{}
body1 := &Body{}
body2 := &Body{}
header1.Extradata = []byte("sender 1btc to alice")
header2.Extradata = []byte("sender 1btc to bob")
//添加兩個(gè)區(qū)塊
bc.AddBlock(header1, body1)
time.Sleep(time.Second)
bc.AddBlock(header2, body2)
//打印鏈中的區(qū)塊信息
for _, block := range bc.blocks {
Logger.Infoln("------------Block", block.header.Number, "--------------------")
Logger.Infoln("prevhash:", block.header.ParentHash)
Logger.Infoln("data:", string(block.header.Extradata))
Logger.Infoln("Blockhash: ", block.header.Blockhash)
Logger.Infoln("blocknumber: ", block.header.Number)
Logger.Infoln("Timestamp: ", timeformat(block.header.Timestamp))
}
}
func timeformat(unixtimestamp int64) string {
time := time.Unix(unixtimestamp, 0)
timeformat := time.Format("Mon Jan _2 15:04:05 MST 2006")
return timeformat
}