IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 区块链 -> POS代码实现 -> 正文阅读

[区块链]POS代码实现

本篇文章主要是POS共识的简单实现,其中有许多地方都做了简化。POS的原理已在上篇文章中描述过,如果对POS的原理不太清晰的可以进行查看。文章地址:共识算法学习总结

代码实现的功能有:添加区块、查看区块链、添加验证者以及查看所有的验证者。代码使用bolt数据库进行数据持久化。

创建一条区块链

系统运行时,首先运行命令创建一条区块链。该函数主要的功能:创建一个创世块,并把创世块保存在bolt数据库中。

func create() {
	// 生成创世块
	genesisBlock := common.Block{}
	genesisBlock = common.Block{
		0,
		time.Now().String(),
		"",
		GenerateHashValue(genesisBlock),
		"",
		common.Validator{},
	}

	// 获取bolt数据库句柄
	db := common.GetDB()
	defer db.Close()
	// 把创世块添加进数据库文件中
	db.Update(func(tx *bolt.Tx) error {

		// 获取bucket
		bucket, err := tx.CreateBucketIfNotExists([]byte(common.BlocksBucket))
		if err != nil{
			fmt.Println(err)
			return err
		}

		// 获取最后一个区块的哈希
		lastHash := bucket.Get([]byte("lastHash"))
		if lastHash == nil {
			bucket.Put([]byte("lastHash"), []byte(genesisBlock.HashCode))
			bucket.Put([]byte(genesisBlock.HashCode), serializeBlock(genesisBlock))
		}
		return err
	})
}

添加一个区块

当创建一个区块链后就可以往区块链中添加区块。当添加区块时首先开启共识模块,共识模块会在后面讲解。开启共识模块后,把输入的数据打包成区块,然后把区块添加到候选区块通道中,等待共识模块的确认。最后等待共识模块的完成。

func addBlock(data string)  {
	// 开始共识
	go consensus.Start()

	// 把客户端输入的数据封装成区块。这里做了简化,直接生成了区块。
	// 实际上应由验证者进行打包区块。
	newBlock := generateBlock(data)

	// 把新生成的区块添加到候选区块通道中
	common.CandidateBlokcs <- newBlock

	select {
	// 阻塞等待退出
	case <- common.ExitChan:
		return
	}
}

启动共识

在启动共识的函数中,首先开启一个协程,不断读取候选区块切片中的数据,如果读取到区块就把区块追加到临时区块切片中。协程启动完毕后就开始循环POS共识。

func Start(){
	// 不断读取候选区块通道,如果有新的区块,就追加到临时区块切片中
	go func() {
		for candidate := range common.CandidateBlokcs{
			fmt.Println("有新的临时区块")
			common.TempBlocks = append(common.TempBlocks, candidate)
		}
	}()


	for  {
		// 循环pos共识算法
		pos()
	}

}

POS共识

进入共识算法后,首先获取临时区块中的区块,接着获取所有的验证者并根据验证者的权重创建一个新的切片。在这个切片中,验证者的权重越高,重复的次数也就越多,也就越容易被选为出块节点。接着通过一个随机数挑选验证者并打包区块。

func pos()  {
	// 复制临时区块
	temp := common.TempBlocks

	// 根据temp的长度判断是否存在临时区块
	if len(temp) > 0{
		var tempValidators []common.Validator
		// 获取所有的验证者
		validators := getAllValidators()
		// 根据验证者拥有的token数量及时间得出权重,权重越高,被选取为出块节点的概率越大
		for i := 0; i < len(validators) - 1; i++{
			for j := 0; j < validators[i].Tokens * validators[i].Days; j++{
				tempValidators = append(tempValidators, validators[i])
			}
		}

		// 获取数据库句柄
		db := common.GetDB()
		defer db.Close()
		for _, block := range temp{
			// 挑选验证者
			rand.Seed(time.Now().Unix())
			var rd =rand.Intn(len(tempValidators))
			block.Validator = validators[rd % len(validators)]
			// 持久化
			db.Update(func(tx *bolt.Tx) error {
				bucket := tx.Bucket([]byte(common.BlocksBucket))
				err := bucket.Put([]byte(block.HashCode), serializeBlock(block))
				if err != nil {
					log.Fatal(err)
				}
				err = bucket.Put([]byte("lastHash"), []byte(block.HashCode))
				if err != nil {
					log.Fatal(err)
				}
				return nil
			})
		}
		mutex.Lock()
		common.ExitChan <- true
		common.TempBlocks = []common.Block{}
		temp = []common.Block{}
		mutex.Unlock()
	}
}

添加验证者

添加验证者时,需要指定验证者的token数量。验证者的地址用时间来模拟。最后把验证者持久化到数据库中,键为验证者的地址,值为序列化后的验证者。

func addValidator(token int)  {
	// 验证者的地址用当前时间的sha256值代替
	address := calculateHash(time.Now().String())

	// 创建一个验证者
	validator := common.Validator{
		token,
		time.Now().Second(),
		address,
	}

	// 添加到数据库中
	db := common.GetDB()
	defer db.Close()
	db.Update(func(tx *bolt.Tx) error {
		bucket := tx.Bucket([]byte(common.PeerBucket))
		err := bucket.Put([]byte(validator.Address), serializeValidator(validator))
		if err != nil {
			log.Fatal(err)
			return err
		}
		return nil
	})
}

查看验证者

查看验证者比较简单,只是查询一下数据库并进行打印。

func view()  {
	var validators []common.Validator

	db := common.GetDB()
	defer db.Close()
	db.View(func(tx *bolt.Tx) error {
		bucket := tx.Bucket([]byte(common.PeerBucket))
		cursor := bucket.Cursor()

		for k, v := cursor.First(); k != nil; k, v = cursor.Next(){
			if string(k) == "lastHash"{
				continue
			}
			block := deserializeValidator(v)
			validators = append(validators, block)
		}
		return nil
	})

	spew.Dump(validators)
}

运行结果

在这里插入图片描述

最后

项目中做了很多简化并且有很多设计不合理的地方,以后会继续进行改进。源码:https://github.com/blockchainGuide/Consensus_Algorithm

  区块链 最新文章
盘点具备盈利潜力的几大加密板块,以及潜在
阅读笔记|让区块空间成为商品,打造Web3云
区块链1.0-比特币的数据结构
Team Finance被黑分析|黑客自建Token“瞒天
区块链≠绿色?波卡或成 Web3“生态环保”标
期货从入门到高深之手动交易系列D1课
以太坊基础---区块验证
进入以太坊合并的五个数字
经典同态加密算法Paillier解读 - 原理、实现
IPFS/Filecoin学习知识科普(四)
上一篇文章      下一篇文章      查看所有文章
加:2022-02-14 21:11:42  更:2022-02-14 21:13:27 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 22:34:32-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码