使用go实现简易比特币区块链公链功能
作者:sgentle 发布时间:2024-04-28 09:18:20
标签:go,比特币,区块链,公链
使用go语言实现具备以下功能的简易区块链
区块与区块链
共识机制
数据库
Cli命令行操作
交易管理
密码学
数字签名
交易缓存池
P2P网络管理
由于平时还要进行论文工作,项目不定时更新
2021.1.1实现了区块结构、区块链结构、工作量证明pow,剩下部分陆续更新
1.实现区块结构
package BLC
import (
"bytes"
"crypto/sha256"
"time"
)
//实现一个最基本的区块结构
type Block struct {
TimeStamp int64 //时间戳,区块产生的时间
Heigth int64//区块高度(索引、号码)代表当前区块的高度
PreBlockHash []byte//前一个区块(父区块)的哈希
Hash []byte//当前区块的哈希
Data []byte//交易数据
}
//创建一个新的区块
func NewBlock(height int64,preBlockHash []byte,Data []byte) *Block {
var block Block
block=Block{Heigth: height,PreBlockHash: preBlockHash,Data: Data,TimeStamp: time.Now().Unix()}
block.SetHash()
return &block
}
//计算区块哈希
func (b *Block)SetHash() {
//int64转换成字节数组
//高度转换
heightBytes:=IntToHex(b.Heigth)
//时间转换
timeStampBytes:=IntToHex(b.TimeStamp)
//拼接所有属性进行hash
blockBytes:=bytes.Join([][]byte{heightBytes,timeStampBytes,b.PreBlockHash,b.Data},[]byte{})
hash:=sha256.Sum256(blockBytes)
b.Hash=hash[:]
}
2.实现区块链结构
package BLC
type BlockChain struct {
Blocks []*Block //存储有序的区块
}
//初始化区块链
func CreateBlockChainWithGenesisBlock() *BlockChain {
//添加创世区块
genesisBlock:=CreateGenesisBlock("the init of blockchain")
return &BlockChain{[]*Block{genesisBlock}}
}
//添加新的区块到区块链中
func (bc *BlockChain)AddBlock(height int64,data []byte,prevBlockHash []byte){
newBlock := NewBlock(height,prevBlockHash,data)
bc.Blocks=append(bc.Blocks,newBlock)
}
3.实现工作量证明
package BLC
import (
"bytes"
"crypto/sha256"
"fmt"
"math/big"
)
//目标难度值,生成的hash前 targetBit 位为0才满足条件
const targetBit =16
//工作量证明
type ProofOfWork struct {
Block *Block //对指定的区块进行验证
target *big.Int //大数据存储
}
//创建新的pow对象
func NewProofOfWork(block *Block) *ProofOfWork {
target:=big.NewInt(1)
target=target.Lsh(target,256-targetBit)
return &ProofOfWork{block,target}
}
//开始工作量证明
func (proofOfWork *ProofOfWork)Run() ([]byte,int64) {
//数据拼接
var nonce=0 //碰撞次数
var hash [32]byte //生成的hash
var hashInt big.Int //存储转换后的hash
for {
dataBytes:=proofOfWork.prepareData(nonce)
hash=sha256.Sum256(dataBytes)
hashInt.SetBytes(hash[:])
fmt.Printf("hash:\r%x",hash)
//难度比较
if proofOfWork.target.Cmp(&hashInt)==1{
break
}
nonce++
}
fmt.Printf("碰撞次数:%d\n",nonce)
return hash[:],int64(nonce)
}
//准备数据,将区块属性拼接起来,返回字节数组
func (pow *ProofOfWork)prepareData(nonce int) []byte {
data:=bytes.Join([][]byte{
pow.Block.PreBlockHash,
pow.Block.Data,
IntToHex(pow.Block.TimeStamp),
IntToHex(pow.Block.Heigth),
IntToHex(int64(nonce)),
IntToHex(targetBit),
},[]byte{})
return data
}
4.当前运行结果
来源:https://blog.csdn.net/sgentle/article/details/112062364


猜你喜欢
- 有时候希望可以用js另开新窗口,但用window.open方法打开窗口总是被浏览器阻止,可以用下面的方法打开新窗口而不会遭到拦截1.新添加一
- sql server 全文检索有两种搜索方式,一种是contains,另一种是freetext。前者是包含,类似于 like '%关
- 今天在网上看到type的一段代码 ,然后查了一下文档,才知道type还有三个参数的用法。http://docs.python.org/2/l
- 一、概述OpenCV在V4.5.3版本的contrib包中提供了一个barcode::BarcodeDetector类,用于条形码的识别。二
- 需要在 ~/.pip/pip.conf 配置文件中加入下面的语句,避免这类警告:没有目录或没有配置文件需要自己新建mkdir ~/.pip/
- 前言使用PyCharm在Python Interpreter设置中的Python虚拟环境安装第三方包时,很有可能报错:Non-zero ex
- 优秀的Stoyan Stefanov在他的新书中(《Javascript Patterns》)介绍了很多编写高质量代码的技巧,比如避免使用全
- 前言在vue项目中我们经常使用到 v-show ,v-if,v-for等内置的指令,除此之外vue还提供了非常方便的自定义指令,供我们对普通
- 安 * oostpython调用C/C++的方法有很多,本文使用boost.python。考虑到后期有好多在boost上的开发工作,所以boo
- 除法啰嗦的,不仅是python。整数除以整数看官请在启动idle之后,练习下面的运算:>>> 2/50>>&g
- 更加颗粒级的缓存框架使用方法是对单个视图的输出进行缓存。 django.views.decorators.cache定义了一个自动缓存视图响
- 实现思路:分为两部分,第一部分,获取网页上数据并使用xlwt生成excel(当然你也可以选择保存到数据库),第二部分获取网页数据使用IO流将
- 前言本文的文字及图片来源于网络,仅供学习、交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理。作者:黑白之道刮刮
- 约定:import pandas as pdimport numpy as npfrom numpy import nan as NaN滤除
- 访问者模式我觉得Visitor模式是在补修改已有程序结构前提下,通过添加额外的访问者完成对代码功能的拓展 为什么这样用?当你的类层次较多,在
- SQL SERVER支持的字符串函数内容:LEN(string)函数LOWER(string)函数UPPER (string)函数LTRIM
- 我们先来看下秒杀活动页面代码<!DOCTYPE HTML><html> <head> <
- get_or_create函数比较好用。如果查询到就返回,如果没查询到就向数据库加入新的对象。e.g.size = Size.objects
- 引言在Go语言中,我们通常会用到panic和recover来抛出错误和捕获错误,这一对操作在单协程环境下我们正常用就好了,并不会踩到什么坑。
- 前言本文提供将图片色彩转为黑白或者褐色风格。比较类似于我们在看动漫、影视作品中,当人物在回忆过程中,体现出来的画面一般都是黑白或者褐色的。环