基本原理这里就不写了,只写一个简单demo的实现

首先得有一个区块用来存储区块头和区块体

type Block struct {
Version int64
PreBlockHash []byte
Hash []byte //区块体内是不存储HASH值的,这是网络中某个节点在计算时存储在息本地的,这里是为了方便实现最终的功能做的
TimeStamp int64
TargetBits int64 //难度值
Nonce int64
MerkelRoot []byte
Data []byte //区块体信息
}

还得有一个方法用于实现区块体

func NewBlock(data string, prevBlockHash []byte) *Block {
//initial block data
block := &Block{
Version:,
PreBlockHash:prevBlockHash,
//Hash:[]
TimeStamp:time.Now().Unix(),
TargetBits:,
Nonce:,
MerkelRoot:[]byte{},
Data:[]byte(data),
}
block.SetHash() //get block hash
return block
}

那区块体的hash值计算就得使用另外一个方法来实现

func (block *Block)SetHash()  {
tmp := [][]byte{
IntToByte(block.Version),
block.PreBlockHash,
IntToByte(block.TimeStamp),
block.MerkelRoot,
IntToByte(block.Nonce),
block.Data,
}
data := bytes.Join(tmp, []byte{})//join接收两个参数,第一个二维数组,第二个这里设置为空的连接符
hash := sha256.Sum256(data)
block.Hash = hash[:]
}

区块有了,那么得一个能实现把区块连接起来的文件,blockchain.go

package main

import "os"

type BlockChain struct {
blocks []*Block
} func NewBlockChain() *BlockChain {
return &BlockChain{[]*Block{NewGensisBlock()}} //添加了创世块
} func (bc *BlockChain)AddBlock(data string) {
if len(bc.blocks) <= {
os.Exit()
}
lastBlock := bc.blocks[len(bc.blocks)-] //还没取呢,如果取要取前一个块的HASH值,即取当前链最后一个块的HASH值
prevBlockHash := lastBlock.Hash
block := NewBlock(data, prevBlockHash)//取前一个块的hash值
bc.blocks = append(bc.blocks, block)
}

上面两个文件涉及到的工具写在另外一个文件

package main

import (
"bytes"
"encoding/binary"
"fmt"
"os"
) func IntToByte(num int64) []byte {
var buffer bytes.Buffer
err := binary.Write(&buffer, binary.BigEndian, num)//将int类型的num使用binary的方法传递到buffer里去
CheckError(err)
return buffer.Bytes()
} func CheckError(err error) {
if err != nil {
fmt.Println("err occur: ", err)
os.Exit()
}
} func NewGensisBlock() *Block { //创建创世块
return NewBlock("Gensis Block!", []byte{})
}

最后主函数的实现

package main

import "fmt"

func main()  {
bc := NewBlockChain{}
bc.AddBlock("班长转给老师一枚BTC")
bc.AddBlock("班长又转给老师一枚BTC")
for i, block := range bc.blocks {//完成区块的遍历
fmt.Println("============= block num : ", i)
fmt.Println("Version", block.Version)
fmt.Printf("PreBlockHash: %x\n", block.PreBlockHash)
fmt.Printf("Hash: %x\n", block.Hash)
fmt.Printf("TimeStamp: %x\n", block.TimeStamp)
fmt.Printf("Nonce: %x\n", block.Nonce)
fmt.Printf("MerkelRoot: %x\n", block.MerkelRoot)
fmt.Printf("Data: %s\n", block.Data)
}
}

[GO]用go语言实现区块链工作原理的更多相关文章

  1. [GO]go语言实现区块链工作证明(pow)原理

    package main import ( "math/big" "bytes" "math" "crypto/sha256&qu ...

  2. Go语言开发区块链只需180行代码

    区块链开发用什么语言?通过本文你将使用Go语言开发自己的区块链(或者说用go语言搭建区块链).理解哈希函数是如何保持区块链的完整性.掌握如何用Go语言编程创造并添加新的块.实现多个节点通过竞争生成块. ...

  3. 比原链CTO James | Go语言成为区块链主流开发语言的四点理由

    11月24日,比原链CTO James参加了Go中国举办的Gopher Meetup杭州站活动,与来自阿里.网易的技术专家带来Kubernetes.区块链.日志采集.云原生等话题的分享.James向大 ...

  4. 用Hyperledger Fabric(超级账本)来构建Java语言开发区块链的环境

    面向 Java 开发人员的链代码简介 您或许听说过区块链,但可能不确定它对 Java™ 开发人员有何用.本教程将帮助大家解惑.我将分步展示如何使用 Hyperledger Fabric v0.6 来构 ...

  5. 使用Go语言编写区块链P2P网络(译)(转)

    转自:https://mp.weixin.qq.com/s/2daFH9Ej5fVlWmpsN5HZzw 外文链接: https://medium.com/coinmonks/code-a-simpl ...

  6. HyperLedger Fabric 1.4 区块链工作过程(2.3)

    区块链的工作过程分交易产生.交易广播.节点计算.获取记账权.记账权广播.接收区块.验证区块和完成记账七个过程. 1) 交易产生:用户向区块链发了一笔交易信息,将产生交易:2) 交易广播:当一笔新交易产 ...

  7. 001/Go语言构建区块链(mooc)

    1.区块链发展与现状 视频地址:https://www.imooc.com/video/17452 注意: 比特币与以太坊最大的区别在于: 以太坊引入了对图灵完美智能合约的支持,人们可以将任何业务逻辑 ...

  8. spring-cloud-Zuul学习(三)【中级篇】--Filter链 工作原理与Zuul原生Filter【重新定义spring cloud实践】

    这里开始记录zuul中级进阶内容.前面说过了,zuul主要是一层一层的Filter过滤器组成,并且Zuul的逻辑引擎与Filter可用其他基于JVM的语言编写,比如:Groovy. 工作原理 Zuul ...

  9. 通过blockchain_go分析区块链交易原理

    原文链接-石匠的Blog 1.背景 在去中心化的区块链中进行交易(转账)是怎么实现的呢?本篇通过blockchain_go来分析一下.需要进行交易,首先就需要有交易的双方以及他们的认证机制,其次是各自 ...

随机推荐

  1. 20181110_wait和async

    一. Awit和async的由来: await/async本身是一个语法糖,编译器提供的一个简化编程的功能; 在C#升级和.net Framework升级的时候, 产生的, 所以说并不是CLR的产物 ...

  2. 第十章 Secret & Configmap (中)

    10.3 在Pod中使用Secret 10.3.1 Volume方式 apiVersion: v1 kind: Pod metaata: name: mypod spec: containers: - ...

  3. VS2015 异常 :遇到异常。这可能是由某个扩展导致的

    原因是安装程序时将注册表修改了,解决方案: 修改注册表: 64位机器: [HKEY_CLASSES_ROOT\CLSID\{73B7DC00-F498-4ABD-AB79-D07AFD52F395}\ ...

  4. 将 .NET 任务作为 WinRT 异步操作公开

    转自:http://blogs.msdn.com/b/windowsappdev_cn/archive/2012/06/22/net-winrt.aspx 在博文深入探究 Await 和 WinRT ...

  5. RAC的时间同步问题

    今天在两个节点上面安装RAC,在安装clusterware的时候OUI总是提示失败.查到资料的得知: 特此记录: 需要在在所有的集群节点上设置正确的日期和时间 在安装 Oracle 集群件.数据库以 ...

  6. cinder backup ceph的配置和使用

    Backup 是将 volume 备份到别的地方(备份设备),将来可以通过 restore 操作恢复. 初看 backup 功能好像与 snapshot 很相似,都可以保存 volume 的当前状态, ...

  7. Cinder Backup备份

    cinder 备份提供了三种驱动服务: Ceph,TSM,Swift 其中默认备份驱动服务为swift cinder 驱动服务的配置在cinder.conf文件中 backup_driver=cind ...

  8. redis 创建集群时 出现的错误解决方式

    1. 创建集群时报以下错误 (1)错误1 ./redis-trib.rb create --replicas 1 XXXXXX:5301 XXXXXX:5302 XXXXXX:5303 XXXXXX: ...

  9. 爬虫中urllib库

    一.urllib库 urllib是Python自带的一个用于爬虫的库,其主要作用就是可以通过代码模拟浏览器发送请求.其常被用到的子模块在Python3中的为urllib.request和urllib. ...

  10. linux系统构架 - LB集群之LVS介绍

    LB 集群是 load balance 集群的简写,翻译成中文就是负载均衡集群.常用的负载均衡开源软件有 nginx.lvs.keepalived ,商业的硬件负载设备 F5.Netscale. LB ...