基本原理这里就不写了,只写一个简单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. scrapy_redis 实现多进程配置部分代码

    # 启用Redis调度存储请求队列SCHEDULER = "scrapy_redis.scheduler.Scheduler"# 确保所有的爬虫通过Redis去重DUPEFILTE ...

  2. centos6.9 x64安装http,php5.6,curl5.29,mysql最后安装zabbix3.4+zabbix客户端

    https://www.zabbix.com/documentation/3.4/zh/manual/installation/requirementshttps://www.zabbix.com/d ...

  3. Hessian简要入门

      原本系统之间通信采用Restful Web Service,但其中没有考虑安全性问题,因此决定使用稍微复杂点的二进制协议,Hessian服务.   Hessian是一个轻量级的Remoting O ...

  4. 模拟admin组件自己开发stark组件之创建篇

    admin组件 admin组件为我们提供了针对django管理页面 我们先简短来看下django的admin组件的启动流程,注册流程,url匹配过程 启动注册 1. 扫描所有应用下的注册了应用中的ad ...

  5. 函数和object

    普通函数 在javascript中,函数是一等公民,函数在javascript是一个数据类型,而非像C#或其他描述性语言那样仅仅作为一个模块来使用. 一.函数调用形式 函数调用形式是最常见的形式,也是 ...

  6. C和指针 第三章--数据

    简要概述: <C和指针>第三章对数据进行了描述. 其中主要讲解了---变量的三个属性:作用域.链接属性和存储类型. 这三个属性决定了该变量在“什么地方可以使用”以及“该变量的值能够保持多久 ...

  7. 用VIM设置UTF-8编码的BOM标记

    1.去掉BOM标记: :set nobomb 2.加上BOM标记: :set bomb 3.查询当前UTF-8编码的文件是否有BOM标记: :set bomb? 4.更高级一点的: :%!xxd &q ...

  8. Swift 延迟运行代码

    // // DelayRun.swift // // Created by XuQing on 16/7/1. // Copyright © 2016年 xuqing. All rights rese ...

  9. Python 迭代器和生成器(转)

    Python 迭代器和生成器 在Python中,很多对象都是可以通过for语句来直接遍历的,例如list.string.dict等等,这些对象都可以被称为可迭代对象.至于说哪些对象是可以被迭代访问的, ...

  10. 如何理解dart的mixin

    mixin翻译出来就是混入的意思 混入,就是一个类可以使用另一个类里的功能比如方法或者属性,其实这个功能并不陌生 ,有点类似c#里的扩展方法,但是并不同于, mixin和implements有着本质的 ...