区块链之bolt数据库持久化与基本功能完善

Posted 一只特立独行的猫

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了区块链之bolt数据库持久化与基本功能完善相关的知识,希望对你有一定的参考价值。

文章目录


链接: 区块链项目github地址
项目目前进度:

bolt数据库安装

bolt数据库介绍:

bolt数据库是一个为go语言提供的轻量化数据集,是一个简单、快速的关系型数据库,其中数据
以[]byte的方式进行存储。索引方式为键/值的方式,即通过关键词去索引数据。

bolt项目地址:link
安装步骤:
1.在项目地址下的终端运行

go get github.com/boltdb/bolt/...

2.当前项目的.mod文件加入语句

require (
github.com/boltdb/bolt v1.3.1 // indirect
golang.org/x/sys v0.3.0 // indirect
)

3.修改mod文件后,需要运行

go mod vendor

进行更新

使用bolt进行持久化存储

相较于之前的区块链程序,使用持久化存储即将区块链存放在数据库中进行保存,进行持久化。相较于之前的未持久化的程序,修改的地方有:
1.将创世区块进行持久化
2.将新增区块进行持久化

bolt持久化的基本步骤

1.创建或打开一个数据库,dbName为数据库的名称,0600说明数据库的权限(0600的二进制码为111,000,000,表示管理员有读、写权限,没有执行权限),nil表示默认操作

	db, err := bolt.Open(dbName, 0600, nil)

2.对数据库进行创建数据表的操作,这里bucket即表示数据表的意思

db.Update(func(tx *bolt.Tx) error 
		b := tx.Bucket([]byte(blockTableName))
		if b == nil 
			//数据表不存在
			b, err := tx.CreateBucket([]byte(blockTableName))
			if err != nil 
				log.Panicf("create backet [%s] failed %v\\n", blockTableName, err)
			
		
		return nil
	)

3.增加与插入数据操作,b.get表示通过键去索引值,得到的值为[]byte的格式,b.put表示存储值,其中,数据必须以键值对的方式进行存储。

bc.DB.Update(func(tx *bolt.Tx) error 
		//获取数据库桶
		b := tx.Bucket([]byte(blockTableName))
		if b != nil 
			blockTypes := b.Get(bc.Tip)
			err = b.Put([]byte("l"), newBlock.Hash)
			if err != nil 
				log.Panicf("save the latest block hash failed %v\\n", err)
			
		
		return nil
	)

创世区块的持久化

原始方式:

func CreateBlockChainWithGenesisBlock() *BlockChain 
	//生成创世区块
	block := CreateGenesisBlock([]byte("the first block"))
	return &BlockChain[]*Blockblock

// 生成创世区块
func CreateGenesisBlock(data []byte) *Block 
	return NewBlock(nil, 1, data)

持久化方式:

db.Update(func(tx *bolt.Tx) error 
		b := tx.Bucket([]byte(blockTableName))
		if b == nil 
			//数据表不存在
			b, err := tx.CreateBucket([]byte(blockTableName))
			if err != nil 
				log.Panicf("create backet [%s] failed %v\\n", blockTableName, err)
			
			//生成创世区块
			genesisBlock := CreateGenesisBlock([]byte("the first block"))
			//键为区块的哈希,值为区块的序列化
			err = b.Put(genesisBlock.Hash, genesisBlock.Serialize())
			if err != nil 
				log.Panicf("insert the genesis block failed %v\\n", err)
			
			blockHash = genesisBlock.Hash
			//数据库中也存储最新区块的哈希
			err = b.Put([]byte("l"), genesisBlock.Hash)
			if err != nil 
				log.Panicf("save the latest hash of genesis block failed %v\\n", err)
			
		 else 
			//数据表已存在
			blockHash = b.Get([]byte("l"))
		
		return nil
	)
	if err != nil 
		log.Panicf("create db [%s] failed %v\\n", dbName, err)
	

新增区块的持久化

原始方式:

func (bc *BlockChain) AddBlock(prevBlockHash []byte, height int64, data []byte) 
	var newBlock *Block
	newBlock = NewBlock(prevBlockHash, height, data)
	bc.Blocks = append(bc.Blocks, newBlock)

持久化方式:

bc.DB.Update(func(tx *bolt.Tx) error 
		//获取数据库桶
		b := tx.Bucket([]byte(blockTableName))
		if b != nil 
			//获取对应区块
			blockTypes := b.Get(bc.Tip)
			//区块结构反序列化
			latestBlock := DeserializeBlock(blockTypes)
			//新建区块,传入参数:prevBlockHash []byte, height int64, data []byte
			newBlock := NewBlock(latestBlock.Hash, latestBlock.Height+1, data)
			//存入数据库
			bc.Tip = newBlock.Hash
			err := b.Put(newBlock.Hash, newBlock.Serialize())
			if err != nil 
				log.Panicf("insert the new block failed %v\\n", err)
			
			err = b.Put([]byte("l"), newBlock.Hash)
			if err != nil 
				log.Panicf("save the latest block hash failed %v\\n", err)
			
		
		return nil
	)

完善区块链基本功能

由于区块链不支持删除、修改,所以当前项目区块链的功能有创建创世区块,增加区块,遍历区块链这三个功能,以后计划完善交易信息、挖矿等多个功能。相较于之前功能的完善有:1.持久化,2.新增遍历功能,3.新增迭代器

创世区块创建

通过将结构体进行序列化,从而使得区块可以通过(区块哈希–>区块信息)的方式存储到数据库中,为后期得到前一区块哈希后进行索引前一区块提供便利。

// 初始化区块链
func CreateBlockChainWithGenesisBlock() *BlockChain 
	//存储最新区块链哈希
	var blockHash []byte
	//1.创建或打开一个数据库
	db, err := bolt.Open(dbName, 0600, nil)
	//2.创建一个桶,将创世区块存入数据库中
	db.Update(func(tx *bolt.Tx) error 
		b := tx.Bucket([]byte(blockTableName))
		if b == nil 
			//数据表不存在
			b, err := tx.CreateBucket([]byte(blockTableName))
			if err != nil 
				log.Panicf("create backet [%s] failed %v\\n", blockTableName, err)
			
			//生成创世区块
			genesisBlock := CreateGenesisBlock([]byte("the first block"))
			//键为区块的哈希,值为区块的序列化
			err = b.Put(genesisBlock.Hash, genesisBlock.Serialize())
			if err != nil 
				log.Panicf("insert the genesis block failed %v\\n", err)
			
			blockHash = genesisBlock.Hash
			//数据库中也存储最新区块的哈希
			err = b.Put([]byte("l"), genesisBlock.Hash)
			if err != nil 
				log.Panicf("save the latest hash of genesis block failed %v\\n", err)
			
		
		return nil
	)
	if err != nil 
		log.Panicf("create db [%s] failed %v\\n", dbName, err)
	
	return &BlockChainDB: db, Tip: blockHash


Block的序列化代码:

// 区块结构序列化,结构体->字节数组
func (block *Block) Serialize() []byte 
	var buffer bytes.Buffer
	//序列化结构体
	encoder := gob.NewEncoder(&buffer)
	err := encoder.Encode(block)
	if err != nil 
		log.Panicf("serialize the block to byte[] failed %v", err)
	
	return buffer.Bytes()

Block的反序列化代码

// 区块结构反序列化,字节数组->结构体
func DeserializeBlock(blockByte []byte) *Block 
	var block Block
	decoder := gob.NewDecoder(bytes.NewReader(blockByte))
	err := decoder.Decode(&block)
	if err != nil 
		log.Panicf("deserialize the byte[] to block failed %v", err)
	
	return &block


增加区块

func (bc *BlockChain) AddBlock(data []byte) 
	//更新区块数据(insert)
	bc.DB.Update(func(tx *bolt.Tx) error 
		//获取数据库桶
		b := tx.Bucket([]byte(blockTableName))
		if b != nil 
			//获取对应区块
			blockTypes := b.Get(bc.Tip)
			//区块结构反序列化
			latestBlock := DeserializeBlock(blockTypes)
			//新建区块,传入参数:prevBlockHash []byte, height int64, data []byte
			newBlock := NewBlock(latestBlock.Hash, latestBlock.Height+1, data)
			//存入数据库
			bc.Tip = newBlock.Hash
			err := b.Put(newBlock.Hash, newBlock.Serialize())
			if err != nil 
				log.Panicf("insert the new block failed %v\\n", err)
			
			err = b.Put([]byte("l"), newBlock.Hash)
			if err != nil 
				log.Panicf("save the latest block hash failed %v\\n", err)
			
		
		return nil
	)

遍历区块链

这里使用了迭代器对区块链进行优化,改善代码可读性。

func (bc *BlockChain) PrintBlockChain() 
	var curBlock *Block
	var iter *BlockChainIterator = bc.Iterator()
	fmt.Printf("打印区块链完整信息。。。\\n")
	//循环读取
	for 
		curBlock = iter.Next()
		fmt.Printf("-------------------------------------------\\n")
		fmt.Printf("\\tHash : %x\\n", curBlock.Hash)
		fmt.Printf("\\tPrevBlockHash : %x\\n", curBlock.PrevBlockHash)
		fmt.Printf("\\tHeight : %x\\n", curBlock.Height)
		fmt.Printf("\\tData : %v\\n", curBlock.Data)
		fmt.Printf("\\tNonce : %x\\n", curBlock.Nonce)
		fmt.Printf("\\tTimeStamp : %x\\n", curBlock.TimeStamp)
		//退出条件,由于哈希值的大数特性,需要使用函数进行比较
		var hashInt big.Int
		hashInt.SetBytes(iter.curHash)
		if big.NewInt(0).Cmp(&hashInt) == 0 
			//遍历到了创世区块
			break
		
	

迭代器的实现如下:

// 实现迭代函数next()
func (iter *BlockChainIterator) Next() *Block 
	var curBlock *Block
	err := iter.DB.View(func(tx *bolt.Tx) error 
		b := tx.Bucket([]byte(blockTableName))
		if b != nil 
			Block := b.Get(iter.curHash)
			curBlock = DeserializeBlock(Block)
			//更新迭代器当前指向结点哈希
			iter.curHash = curBlock.PrevBlockHash
		
		return nil
	)
	if err != nil 
		log.Panicf("Iterator the db failed%v\\n", err)
	 else 
		return curBlock
	
	return nil


以上是关于区块链之bolt数据库持久化与基本功能完善的主要内容,如果未能解决你的问题,请参考以下文章

自己动手写区块链之交易

创建区块链实现之v3本地持久化(bolt数据库的序列化和反序列化)和命令行参数

:自己动手写区块链之最小可行区块链

从0到1简易区块链开发手册V0.3-数据持久化与创世区块

区块链之开发命令行操作模块

区块链之开发命令行操作模块