目錄
專注于為中小企業(yè)提供成都做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)服務(wù),電腦端+手機(jī)端+微信端的三站合一,更高效的管理,為中小企業(yè)清河門免費(fèi)做網(wǎng)站提供優(yōu)質(zhì)的服務(wù)。我們立足成都,凝聚了一批互聯(lián)網(wǎng)行業(yè)人才,有力地推動(dòng)了千余家企業(yè)的穩(wěn)健成長,幫助中小企業(yè)通過網(wǎng)站建設(shè)實(shí)現(xiàn)規(guī)模擴(kuò)充和轉(zhuǎn)變。一. flag基本使用
通常我們?cè)趯懨钚谐绦颍üぞ?、server)時(shí),對(duì)命令參數(shù)進(jìn)行解析是常見的需求。各種語言一般都會(huì)提供解析命令行參數(shù)的方法或庫,以方便程序員使用。在 go 標(biāo)準(zhǔn)庫中提供了一個(gè)包:flag,方便進(jìn)行命令行解析。
1.導(dǎo)入flag包
import (
"flag"
)
2.使用示例
//定義一個(gè)字符串flag,flag名為printchain,默認(rèn)值為:hello BTC world,參數(shù)說明: 輸出所有的區(qū)塊信息
flagString := flag.String("printchain", "hello BTC world", "輸出所有的區(qū)塊信息")
//定義一個(gè)整型flag,flag名為:number ,默認(rèn)值為:6 ,參數(shù)說明:輸入一個(gè)整數(shù)
flagInt := flag.Int("number", 6, "輸出一個(gè)整數(shù)...")
//定義一個(gè)布爾類型的flag,flag名為:open,默認(rèn)值:false,參數(shù)說明:判斷真假
flagBool := flag.Bool("open", false, "判斷真假...")
//解析flag
flag.Parse()
//輸入?yún)?shù)后的值
fmt.Printf("%s\n", *flagString)
fmt.Printf("%d\n", *flagInt)
fmt.Printf("%t\n", *flagBool)
執(zhí)行結(jié)果
//編譯main.go
go build main.go
2.1 默認(rèn)執(zhí)行
./main //執(zhí)行./main,輸出所有flag的默認(rèn)值
hello BTC world
6
false
2.2 .輸入一個(gè)未定義的flag
./main -xx //xx并未定義,輸出錯(cuò)誤信息以及所有flag使用方法(通過flag.Parse()生效)
flag provided but not defined: -xx
Usage of ./main:
-number int
輸出一個(gè)整數(shù)... (default 6)
-open
判斷真假...
-printchain string
輸出所有的區(qū)塊信息 (default "hello BTC world")
2.3 輸入已定義的flag
./main -printchain //僅僅傳入flag,提示flag需要一個(gè)參數(shù)值
flag needs an argument: -printchain
Usage of ./main:
-number int
輸出一個(gè)整數(shù)... (default 6)
-open
判斷真假...
-printchain string
輸出所有的區(qū)塊信息 (default "hello BTC world")
2.4 輸入已定義的flag以及對(duì)應(yīng)類型的值
./main -printchain "hello bruce" //傳入flag=printchain,value="hello bruce"
hello bruce
6
false
./main -number 88 //傳入flag=number,value=88
hello BTC world
88
false
./main -open true //傳入flag=open,value=true (默認(rèn)為false,傳入-open即為true)
hello BTC world
6
true
二.os.Args基本使用
os包提供了一些與操作系統(tǒng)交互的函數(shù)和變量,并且go對(duì)其做了一些封裝。程序的命令行參數(shù)可以從os包的Args變量獲?。籵s包外部使用os.Args訪問該變量。
Go言里也采用左閉右開形式, 即,區(qū)間包括第一個(gè)索引元素,不包括最后一個(gè), 因?yàn)檫@樣可以簡(jiǎn)化邏輯。os.Args的第一個(gè)元素,os.Args[0], 是命令本身的名字;其它的元素則是程序啟動(dòng)時(shí)傳給它的參數(shù)。
1.導(dǎo)入os包
import os
使用示例
//實(shí)例化os.Args
args := os.Args
//打印args切片所有內(nèi)容
fmt.Printf("%v\n",args)
//打印args切片的第二個(gè)參數(shù)
fmt.Printf("%v\n",args[1])
//打印args切片的第三個(gè)參數(shù)
fmt.Printf("%v\n",args[2])
傳入?yún)?shù)
./main arg1 arg2 //執(zhí)行main程序,傳入兩個(gè)參數(shù)arg
[./main arg1 arg2] //args
arg1 //args[1]
arg2 //args[2]
三.flag與os.Args組合使用
1.創(chuàng)建flag對(duì)象
//通過flag.NewFlagSet實(shí)例化flag對(duì)象
addBlockCmd := flag.NewFlagSet("addBlock", flag.ExitOnError)
printChainCmd := flag.NewFlagSet("printChain", flag.ExitOnError)
2.拼接flag
flag名稱:data
默認(rèn)值:bruce
使用說明:交易數(shù)據(jù)
使用方法: ./main addBlock -data
flagAddBlockData := addBlockCmd.String("data", "bruce", "交易數(shù)據(jù)")
3.判斷flag
//判斷os.Args中第二個(gè)flag的值
switch os.Args[1] {
case "addBlock":
err := addBlockCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
case "printChain":
err := printChainCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
default:
printUsage()
os.Exit(1)
}
4.解析flag輸出信息
if addBlockCmd.Parsed() {
if *flagAddBlockData == "" {
printUsage()
os.Exit(1)
}
fmt.Println(*flagAddBlockData)
}
if printChainCmd.Parsed() {
fmt.Println("測(cè)試輸出區(qū)塊信息...")
}
5.相關(guān)函數(shù)
5.1 打印使用說明
func printUsage() {
fmt.Println("Usage:")
fmt.Println("\taddblock -data DATA -- 交易數(shù)據(jù)")
fmt.Println("\tprintChain --輸出區(qū)塊信息")
}
5.2 判斷是否輸入flag
func isValid() {
//如果args切片長度小于2(即沒有跟參數(shù),則打印使用方法并退出程序)
if len(os.Args) < 2 {
printUsage()
os.Exit(1)
}
}
6.測(cè)試命令行
6.1 無參數(shù)
./main
Usage:
addBlock -data DATA -- 交易數(shù)據(jù):
printChain --輸出區(qū)塊信息
6.2 無效參數(shù)
./main xxx
Usage:
addBlock -data DATA -- 交易數(shù)據(jù):
printChain --輸出區(qū)塊信息
6.3 參數(shù)默認(rèn)值
./main addBlock //輸出flag=data的默認(rèn)值bruce
bruce
./main printChain //輸出解析到printChain后打印的信息
測(cè)試輸出區(qū)塊信息...
6.4 指定參數(shù)
./main addBlock -data "brucefeng" //輸出flag=data的輸入值brucefeng
brucefeng
四.通過命令行添加/查詢區(qū)塊
1.實(shí)現(xiàn)目標(biāo)
Usage:
createblockchain -data DATA --交易數(shù)據(jù)
addBlock -data DATA -- 交易數(shù)據(jù)
printChain --輸出區(qū)塊信息
2.定義命令行屬性與方法 CLI.go
2.1 導(dǎo)入相關(guān)包
import (
"fmt"
"os"
"flag"
"log"
)
2.2 定義結(jié)構(gòu)體
type CLI struct{}
2.3 打印幫助提示
func printUsage() {
fmt.Println("Usage:")
fmt.Println("\tcreateblockchain -data DATA --交易數(shù)據(jù)")
fmt.Println("\taddBlock -data DATA -- 交易數(shù)據(jù)")
fmt.Println("\tprintChain --輸出區(qū)塊信息")
}
2.4 判斷參數(shù)輸入是否合法
func isValid() {
if len(os.Args) < 2 {
printUsage()
os.Exit(1)
}
}
2.5 定義創(chuàng)世區(qū)塊的方法
func (cli *CLI) createGenenisBlockChain(data string) {
CreateBlockChainWithGenesisBlock(data)
}
2.6 定義普通區(qū)塊的方法
func (cli *CLI) addBlock(data string) {
//判斷數(shù)據(jù)庫是否存在
if !DBExists() {
fmt.Println("當(dāng)前不存在區(qū)塊鏈,請(qǐng)先創(chuàng)建創(chuàng)世區(qū)塊")
os.Exit(1)
}
//通過BlockChainObject()函數(shù)獲取一個(gè)block
blockchain := BlockChainObject()
defer blockchain.DB.Close()
//通過blockchain的AddBlockChain方法添加區(qū)塊
blockchain.AddBlockChain(data)
}
2.7 定義遍歷區(qū)塊的方法
func (cli *CLI) printChain() {
//判斷數(shù)據(jù)庫是否存在
if !DBExists() {
fmt.Println("當(dāng)前不存在區(qū)塊鏈")
os.Exit(1)
}
blockchain := BlockChainObject()
defer blockchain.DB.Close()
blockchain.PrintChain()
}
2.8 定義命令行參數(shù)方法集合
func (cli *CLI) Run() {
//1.判斷命令行參數(shù)是否合法
isValid()
//2.通過flag.NewFlagSet實(shí)例化三個(gè)flag對(duì)象
//創(chuàng)建創(chuàng)世區(qū)塊
createBlockCmd := flag.NewFlagSet("createblockchain", flag.ExitOnError)
//創(chuàng)建普通區(qū)塊
addBlockCmd := flag.NewFlagSet("addBlock", flag.ExitOnError)
//遍歷打印區(qū)塊信息
printChainCmd := flag.NewFlagSet("printChain", flag.ExitOnError)
//3.定義命令行對(duì)象的相關(guān)屬性
flagCreateBlockChainWithData := createBlockCmd.String("data", "", "交易數(shù)據(jù)")
flagAddBlockData := addBlockCmd.String("data", "", "交易數(shù)據(jù)")
//4.根據(jù)參數(shù)值決定進(jìn)行解析的信息
switch os.Args[1] {
case "createblockchain":
err := createBlockCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
case "addBlock":
err := addBlockCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
case "printChain":
err := printChainCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
default:
printUsage()
os.Exit(1)
}
//5. 判斷是否解析成功 Parsed reports whether f.Parse has been called.
//5.1 如果createBlockCmd解析成功,輸出的DATA的值作為創(chuàng)建創(chuàng)世區(qū)塊的傳入?yún)?shù)
if createBlockCmd.Parsed() {
if *flagCreateBlockChainWithData == "" {
fmt.Println("創(chuàng)世區(qū)塊交易數(shù)據(jù)不能為空")
printUsage()
os.Exit(1)
}
//調(diào)用createGenenisBlockChain()方法創(chuàng)建創(chuàng)世區(qū)塊
cli.createGenenisBlockChain(*flatCreateBlockChainWithData)
}
//5.2 如果addBlockCmd解析成功,輸出的DATA的值作為創(chuàng)建新區(qū)塊的傳入?yún)?shù)
if addBlockCmd.Parsed() {
if *flagAddBlockData == "" {
fmt.Println("創(chuàng)建區(qū)塊交易數(shù)據(jù)不能為空")
printUsage()
os.Exit(1)
}
//調(diào)用addBlock方法創(chuàng)建區(qū)塊
cli.addBlock(*flagAddBlockData)
}
//5.3 如果printChainCmd解析成功,調(diào)用遍歷區(qū)塊鏈的方法
if printChainCmd.Parsed() {
cli.printChain()
}
}
2.9 代碼整合
package BLC
import (
"fmt"
"os"
"flag"
"log"
)
type CLI struct{}
func printUsage() {
fmt.Println("Usage:")
fmt.Println("\tcreateblockchain -data DATA --交易數(shù)據(jù)")
fmt.Println("\taddBlock -data DATA -- 交易數(shù)據(jù)")
fmt.Println("\tprintChain --輸出區(qū)塊信息")
}
func isValid() {
if len(os.Args) < 2 {
printUsage()
os.Exit(1)
}
}
func (cli *CLI) addBlock(data string) {
if !DBExists() {
fmt.Println("數(shù)據(jù)庫不存在")
os.Exit(1)
}
blockchain := BlockChainObject()
defer blockchain.DB.Close()
blockchain.AddBlockChain(data)
}
func (cli *CLI) printChain() {
if !DBExists() {
fmt.Println("數(shù)據(jù)庫不存在")
os.Exit(1)
}
blockchain := BlockChainObject()
defer blockchain.DB.Close()
blockchain.PrintChain()
}
func (cli *CLI) createGenenisBlockChain(data string) {
CreateBlockChainWithGenesisBlock(data)
}
func (cli *CLI) Run() {
isValid()
addBlockCmd := flag.NewFlagSet("addBlock", flag.ExitOnError)
printChainCmd := flag.NewFlagSet("printChain", flag.ExitOnError)
createBlockCmd := flag.NewFlagSet("createblockchain", flag.ExitOnError)
flagAddBlockData := addBlockCmd.String("data", "", "交易數(shù)據(jù)")
flagCreateBlockChainWithData := createBlockCmd.String("data", "", "交易數(shù)據(jù)")
switch os.Args[1] {
case "addBlock":
err := addBlockCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
case "printChain":
err := printChainCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
case "createblockchain":
err := createBlockCmd.Parse(os.Args[2:])
if err != nil {
log.Panic(err)
}
default:
printUsage()
os.Exit(1)
}
if addBlockCmd.Parsed() {
if *flagAddBlockData == "" {
printUsage()
os.Exit(1)
}
cli.addBlock(*flagAddBlockData)
}
if printChainCmd.Parsed() {
cli.printChain()
}
if createBlockCmd.Parsed() {
if *flagCreateBlockChainWithData == "" {
fmt.Println("交易數(shù)據(jù)不能為空")
printUsage()
os.Exit(1)
}
cli.createGenenisBlockChain(*flagCreateBlockChainWithData)
}
}
3.生成區(qū)塊鏈方法改造 BlockChain.go
3.1 定義判斷數(shù)據(jù)庫是否存在的方法
func DBExists() bool {
if _, err := os.Stat(dbName); os.IsNotExist(err) {
return false
}
return true
}
3.2 定義返回BlockChain對(duì)象
func BlockChainObject() *BlockChain {
//定義tip用于存儲(chǔ)從數(shù)據(jù)庫中獲取到的最新區(qū)塊的Hash值
var tip []byte
//打開數(shù)據(jù)庫
db, err := bolt.Open(dbName, 0600, nil)
if err != nil {
log.Fatal(err)
}
//通過key="l"獲取value值(最新區(qū)塊的Hash值)
err = db.View(func(tx *bolt.Tx) error {
//獲取表對(duì)象
b := tx.Bucket([]byte(blockTableName))
if b != nil {
//獲取最新區(qū)塊的Hash值
tip = b.Get([]byte("l"))
}
return nil
})
//返回保存最新區(qū)塊Hash信息的BlockChain對(duì)象
return &BlockChain{tip, db}
}
3.3 創(chuàng)建帶有創(chuàng)世區(qū)塊的區(qū)塊鏈
func CreateBlockChainWithGenesisBlock(data string) {
//判斷數(shù)據(jù)庫是否存在
if DBExists() {
fmt.Println("創(chuàng)世區(qū)塊已經(jīng)存在")
os.Exit(1)
}
//打開數(shù)據(jù)庫
db, err := bolt.Open(dbName, 0600, nil)
if err != nil {
log.Fatal(err)
}
err = db.Update(func(tx *bolt.Tx) error {
//創(chuàng)建數(shù)據(jù)庫表
b, err := tx.CreateBucket([]byte(blockTableName))
if err != nil {
log.Panic(err)
}
if b != nil {
//創(chuàng)建創(chuàng)世區(qū)塊
genesisBlock := CreateGenesisBlock(data)
//將創(chuàng)世區(qū)塊存儲(chǔ)至表中
err := b.Put(genesisBlock.Hash, genesisBlock.Serialize())
if err != nil {
log.Panic(err)
}
//存儲(chǔ)最新的區(qū)塊鏈的hash
err = b.Put([]byte("l"), genesisBlock.Hash)
if err != nil {
log.Panic(err)
}
}
return nil
})
if err != nil {
log.Fatal(err)
}
}
3.4 創(chuàng)建添加區(qū)塊的方法
func (blc *BlockChain) AddBlockChain(data string) {
err := blc.DB.Update(func(tx *bolt.Tx) error {
//1.獲取表
b := tx.Bucket([]byte(blockTableName))
//2.創(chuàng)建新區(qū)塊
if b != nil {
//獲取最新區(qū)塊
byteBytes := b.Get(blc.Tip)
//反序列化
block := DeserializeBlock(byteBytes)
//3. 將區(qū)塊序列化并且存儲(chǔ)到數(shù)據(jù)庫中
newBlock := NewBlock(data, block.Height+1, block.Hash)
err := b.Put(newBlock.Hash, newBlock.Serialize())
if err != nil {
log.Panic(err)
}
//4.更新數(shù)據(jù)庫中"l"對(duì)應(yīng)的Hash
err = b.Put([]byte("l"), newBlock.Hash)
if err != nil {
log.Panic(err)
}
//5. 更新blockchain的Tip
blc.Tip = newBlock.Hash
}
return nil
})
if err != nil {
log.Panic(err)
}
}
3.5 遍歷區(qū)塊鏈的方法
func (blc *BlockChain) PrintChain() {
blockchainIterator := blc.Iterator()
for {
block := blockchainIterator.Next()
fmt.Printf("Height:%d\n", block.Height)
fmt.Printf("PreBlockHash:%x\n", block.PreBlockHash)
fmt.Printf("Data:%s\n", block.Data)
fmt.Printf("TimeStamp:%s\n", time.Unix(block.TimeStamp, 0).Format("2006-01-02 03:04:05 PM"))
fmt.Printf("Hash:%x\n", block.Hash)
fmt.Printf("Nonce:%d\n", block.Nonce)
var hashInt big.Int
hashInt.SetBytes(block.PreBlockHash)
if big.NewInt(0).Cmp(&hashInt) == 0 {
break
}
}
}
3.6 代碼整合
package BLC
import (
"github.com/boltdb/bolt"
"log"
"math/big"
"fmt"
"time"
"os"
)
const dbName = "blockchain.db" //數(shù)據(jù)庫名
const blockTableName = "blocks" //表名
type BlockChain struct {
Tip []byte //區(qū)塊鏈里面最后一個(gè)區(qū)塊的Hash
DB *bolt.DB //數(shù)據(jù)庫
}
//迭代器
func (blockchain *BlockChain) Iterator() *BlockChainIterator {
return &BlockChainIterator{blockchain.Tip, blockchain.DB}
}
//判斷數(shù)據(jù)庫是否存在
func DBExists() bool {
if _, err := os.Stat(dbName); os.IsNotExist(err) {
return false
}
return true
}
func (blc *BlockChain) PrintChain() {
blockchainIterator := blc.Iterator()
for {
block := blockchainIterator.Next()
fmt.Printf("Height:%d\n", block.Height)
fmt.Printf("PreBlockHash:%x\n", block.PreBlockHash)
fmt.Printf("Data:%s\n", block.Data)
fmt.Printf("TimeStamp:%s\n", time.Unix(block.TimeStamp, 0).Format("2006-01-02 03:04:05 PM"))
fmt.Printf("Hash:%x\n", block.Hash)
fmt.Printf("Nonce:%d\n", block.Nonce)
var hashInt big.Int
hashInt.SetBytes(block.PreBlockHash)
if big.NewInt(0).Cmp(&hashInt) == 0 {
break
}
}
}
//
func (blc *BlockChain) AddBlockChain(data string) {
err := blc.DB.Update(func(tx *bolt.Tx) error {
//1.獲取表
b := tx.Bucket([]byte(blockTableName))
//2.創(chuàng)建新區(qū)塊
if b != nil {
//獲取最新區(qū)塊
byteBytes := b.Get(blc.Tip)
//反序列化
block := DeserializeBlock(byteBytes)
//3. 將區(qū)塊序列化并且存儲(chǔ)到數(shù)據(jù)庫中
newBlock := NewBlock(data, block.Height+1, block.Hash)
err := b.Put(newBlock.Hash, newBlock.Serialize())
if err != nil {
log.Panic(err)
}
//4.更新數(shù)據(jù)庫中"l"對(duì)應(yīng)的Hash
err = b.Put([]byte("l"), newBlock.Hash)
if err != nil {
log.Panic(err)
}
//5. 更新blockchain的Tip
blc.Tip = newBlock.Hash
}
return nil
})
if err != nil {
log.Panic(err)
}
}
//1.創(chuàng)建帶有創(chuàng)世區(qū)塊的區(qū)塊鏈
func CreateBlockChainWithGenesisBlock(data string) {
//判斷數(shù)據(jù)庫是否存在
if DBExists() {
fmt.Println("創(chuàng)世區(qū)塊已經(jīng)存在")
os.Exit(1)
}
//打開數(shù)據(jù)庫
db, err := bolt.Open(dbName, 0600, nil)
if err != nil {
log.Fatal(err)
}
err = db.Update(func(tx *bolt.Tx) error {
//創(chuàng)建數(shù)據(jù)庫表
b, err := tx.CreateBucket([]byte(blockTableName))
if err != nil {
log.Panic(err)
}
if b != nil {
//創(chuàng)建創(chuàng)世區(qū)塊
genesisBlock := CreateGenesisBlock(data)
//將創(chuàng)世區(qū)塊存儲(chǔ)至表中
err := b.Put(genesisBlock.Hash, genesisBlock.Serialize())
if err != nil {
log.Panic(err)
}
//存儲(chǔ)最新的區(qū)塊鏈的hash
err = b.Put([]byte("l"), genesisBlock.Hash)
if err != nil {
log.Panic(err)
}
}
return nil
})
if err != nil {
log.Fatal(err)
}
}
//返回BlockChain對(duì)象
func BlockChainObject() *BlockChain {
var tip []byte
//打開數(shù)據(jù)庫
db, err := bolt.Open(dbName, 0600, nil)
if err != nil {
log.Fatal(err)
}
err = db.View(func(tx *bolt.Tx) error {
b := tx.Bucket([]byte(blockTableName))
if b != nil {
//讀取最新區(qū)塊的Hash
tip = b.Get([]byte("l"))
}
return nil
})
return &BlockChain{tip, db}
}
以上未改造代碼可以參考:【Golang區(qū)塊鏈開發(fā)003】區(qū)塊序列化存儲(chǔ)
https://mp.weixin.qq.com/s?__biz=MzA4Mzg5NTEyOA==&mid=2651781600&idx=1&sn=ce72dd45079759dd004be83ed3f9f90c&chksm=84152fd7b362a6c18d1b752b8ffc14d07a057dd4f263c14c3058a822a7b74f7ae7bcb9eb1cd2#rd
五.測(cè)試代碼與測(cè)試結(jié)果
測(cè)試代碼main.go
func main() {
//初始化CLI對(duì)象
cli := BLC.CLI{}
fmt.Println("====返回結(jié)果======")
//執(zhí)行Run命令調(diào)用對(duì)應(yīng)參數(shù)對(duì)應(yīng)的方法
cli.Run()
}
2.測(cè)試結(jié)果
2.1 無參數(shù)執(zhí)行
./main //直接執(zhí)行
====返回結(jié)果======
Usage:
createblockchain -data DATA --交易數(shù)據(jù)
addBlock -data DATA -- 交易數(shù)據(jù)
printChain --輸出區(qū)塊信息
2.2 首次遍歷區(qū)塊鏈
./main printChain
====返回結(jié)果======
當(dāng)前不存在區(qū)塊鏈
2.3 首次添加普通區(qū)塊
./main addBlock 或者 ./main addBlock -data "bruce"
====返回結(jié)果======
當(dāng)前不存在區(qū)塊鏈,請(qǐng)先創(chuàng)建創(chuàng)世區(qū)塊
2.4 創(chuàng)建創(chuàng)世區(qū)塊
./main createblockchain --data "Create Genenis Block 20180708"
====返回結(jié)果======
第1個(gè)區(qū)塊,挖礦成功:0000d5b050b448b6db0d273b9320036f93b7cb2e8f8005c1e6192dc91b4a3381
2018-07-08 22:03:23.939771259 +0800 CST m=+0.087935424
2.5 遍歷區(qū)塊鏈
./main printChain
====返回結(jié)果======
Height:1
PreBlockHash:0000000000000000000000000000000000000000000000000000000000000000
Data:Create Genenis Block 20180708
TimeStamp:2018-07-08 10:03:23 PM
Hash:0000d5b050b448b6db0d273b9320036f93b7cb2e8f8005c1e6192dc91b4a3381
Nonce:61212
2.6 添加區(qū)塊
./main addBlock -data "bruce"
====返回結(jié)果======
第2個(gè)區(qū)塊,挖礦成功:000081872fe39a35be79e30f915c9716b9bbbae74f665fdd53f4ab57ae4b379d
2018-07-08 22:06:27.642330532 +0800 CST m=+0.055636890
2.7 遍歷區(qū)塊鏈
./main printChain
====返回結(jié)果======
Height:2
PreBlockHash:0000d5b050b448b6db0d273b9320036f93b7cb2e8f8005c1e6192dc91b4a3381
Data:bruce
TimeStamp:2018-07-08 10:06:27 PM
Hash:000081872fe39a35be79e30f915c9716b9bbbae74f665fdd53f4ab57ae4b379d
Nonce:39906
Height:1
PreBlockHash:0000000000000000000000000000000000000000000000000000000000000000
Data:Create Genenis Block 20180708
TimeStamp:2018-07-08 10:03:23 PM
Hash:0000d5b050b448b6db0d273b9320036f93b7cb2e8f8005c1e6192dc91b4a3381
Nonce:61212
2.8 添加多個(gè)區(qū)塊
./main addBlock -data "bruce"
./main addBlock -data "send 100 BTC TO bruce"
......
第3個(gè)區(qū)塊,挖礦成功:0000565d60b6a26b8fc238bfeffb2cc1de20d28ca2a312bef9601997bb914fc3
2018-07-08 22:09:52.644353193 +0800 CST m=+0.032078853
第4個(gè)區(qū)塊,挖礦成功:0000dd1354cc868b96aeb18360b320b7a30ea0c00b27b79c32c0d9269ff2dbb3
2018-07-08 22:10:22.805477297 +0800 CST m=+0.210806453
2.9 遍歷整個(gè)區(qū)塊鏈
./main printChain
====返回結(jié)果======
Height:6
PreBlockHash:000019ba65a19b26f81fae22025e963e68a6cf0983fb211aae208c428368252d
Data:bruce send 30 BTC TO jackma
TimeStamp:2018-07-08 10:12:57 PM
Hash:0000e3ba0e34ace19cba574deefdf5b75315b2ce10e488332952544e5a056571
Nonce:3502
Height:5
PreBlockHash:0000dd1354cc868b96aeb18360b320b7a30ea0c00b27b79c32c0d9269ff2dbb3
Data:bruce send 50 BTC TO ponyma
TimeStamp:2018-07-08 10:12:44 PM
Hash:000019ba65a19b26f81fae22025e963e68a6cf0983fb211aae208c428368252d
Nonce:80833
Height:4
PreBlockHash:0000565d60b6a26b8fc238bfeffb2cc1de20d28ca2a312bef9601997bb914fc3
Data:send 100 BTC TO bruce
TimeStamp:2018-07-08 10:10:22 PM
Hash:0000dd1354cc868b96aeb18360b320b7a30ea0c00b27b79c32c0d9269ff2dbb3
Nonce:182779
Height:3
PreBlockHash:000081872fe39a35be79e30f915c9716b9bbbae74f665fdd53f4ab57ae4b379d
Data:bruce
TimeStamp:2018-07-08 10:09:52 PM
Hash:0000565d60b6a26b8fc238bfeffb2cc1de20d28ca2a312bef9601997bb914fc3
Nonce:21138
Height:2
PreBlockHash:0000d5b050b448b6db0d273b9320036f93b7cb2e8f8005c1e6192dc91b4a3381
Data:bruce
TimeStamp:2018-07-08 10:06:27 PM
Hash:000081872fe39a35be79e30f915c9716b9bbbae74f665fdd53f4ab57ae4b379d
Nonce:39906
Height:1
PreBlockHash:0000000000000000000000000000000000000000000000000000000000000000
Data:Create Genenis Block 20180708
TimeStamp:2018-07-08 10:03:23 PM
Hash:0000d5b050b448b6db0d273b9320036f93b7cb2e8f8005c1e6192dc91b4a3381
Nonce:61212
2.10 測(cè)試重新創(chuàng)建區(qū)塊結(jié)構(gòu)
./main createblockchain -data "第二次創(chuàng)建創(chuàng)世區(qū)塊"
====返回結(jié)果======
創(chuàng)世區(qū)塊已經(jīng)存在
創(chuàng)新互聯(lián)www.cdcxhl.cn,專業(yè)提供香港、美國云服務(wù)器,動(dòng)態(tài)BGP最優(yōu)骨干路由自動(dòng)選擇,持續(xù)穩(wěn)定高效的網(wǎng)絡(luò)助力業(yè)務(wù)部署。公司持有工信部辦法的idc、isp許可證, 機(jī)房獨(dú)有T級(jí)流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確進(jìn)行流量調(diào)度,確保服務(wù)器高可用性。佳節(jié)活動(dòng)現(xiàn)已開啟,新人活動(dòng)云服務(wù)器買多久送多久。