/*

作 者: itdef 
欢迎转帖 请保持文本完整并注明出处 
技术博客 http://www.cnblogs.com/itdef/ 
技术交流群 群号码:432336863
欢迎c c++ windows驱动爱好者 服务器程序员沟通交流
部分老代码存放地点
http://www.oschina.net/code/list_by_user?id=614253

*/

本章节相比前两节课程,增加了交易等处理,而在工作量证明,挖矿,创建区块,以及区块链的操作也细致了不少。

工程的具体建立可以参考前两章节内容,这里不再赘述。

区块Block

我们在工程中首先创建Block.h和Block.cpp,用来实现block区块的功能

我们不会实现一个如同比特币技术中使用的区块链,那太复杂。我们实现一个简化版的区块链基本原形,它仅包含了一些创建时间,区块描述以及本身的哈希数值,以及上一个区块的哈希数值。

 class Block {
public:
string _hash; //当前区块的哈希
string _data; //区块描述字符
string _prevHash; //记录上个块的哈希值
Block(const string& prevHash, const string& dataIn); //构造函数
void SetHash(); //设置本区块的哈希
private:
int64_t _nNonce; //区块随机数 用于哈希值的产生
time_t _tTime; //创建时间 };
SetHash()函数就是根据区块的这些属性计算出区块的哈希值。计算出符合标准的区块哈希值应该是一个复杂困难的过程,即使在高速电脑中,也会人为的提升阀值,让区块的产生不那么快速。这个在随后的章节会介绍,这里只是使用最基本的计算方法。
根据结构体中的区块描述字符,创建时间以及上个区块的哈希值等元素计算出哈希值。 区块描述字符和创建时间是让哈希的计算有随机性,而添加上个区块的哈希值是让所有区块都具有关联性,用来提升篡改区块信息的难度。
代码中使用的sha256函数来自来自Zedwood的C++ sha256函数,具体介绍可以查看前面两个章节
 void Block::SetHash() {
stringstream ss;
ss << _tTime << _data << _prevHash;
_hash = sha256(ss.str());
}

区块类的创建函数很简单就是填写各类信息,计算该区块的哈希值

Block::Block( const string& dataIn, const string&    prevHash) {
_tTime = time(nullptr);
_nNonce = -;
_data = dataIn;
_prevHash = prevHash;
SetHash();
}

区块链Blockchain

区块链就是一个个区块的集合。使用vector<Block*> blocks 来存储区块的指针.

与go语言不同的是 所有的内存分配,我们需要自行在退出前归还内存。所以在析构函数中依次遍历容器内的指针并且进行删除内存。结构体如下

class Blockchain {
public:
Blockchain(Block* p);
vector<Block*> blocks;
void AddBlock(string datain);
~Blockchain() {
for (int i = ; i < blocks.size(); i++) {
if (blocks[i] != NULL) {
delete blocks[i];
blocks[i] = NULL;
}
}
}
private: };

区块链创建时候会添加一个Genesis Block创世区块。 这个区块与其他区块的区别在于该区块没有上一个区块信息,它是第一个区块。

        static Block* NewGenesisBlock() {
return NewBlock("Genesis Block", "");
} static Blockchain* NewBlockchain() {
//创建一个创世块 创世块没有prevhash 它是第一个块
Block* pblock = NewGenesisBlock();
Blockchain* p = new Blockchain(pblock);
return p;
}
static Block* NewBlock(string datain, string prevBlockHash) {
Block* p = new Block( datain, prevBlockHash);
return p;
}
Blockchain::Blockchain(Block* p) {
blocks.clear();
blocks.push_back(p);
}

下面开始在main函数检测我们的函数是否工作正常

#include "Blockchain.h"
#include "util.h"
#include <vector>
#include <iostream> using namespace std; int main()
{
Blockchain* bc = TOOLS::NewBlockchain(); bc->AddBlock("Send 1 BTC to Ivan");
bc->AddBlock("Send 2 more BTC to Ivan"); for (int i = ; i < bc->blocks.size(); i++) {
std::cout << "Prev hash = " << bc->blocks[i]->_prevHash << std::endl;
std::cout << "data = " << bc->blocks[i]->_data << std::endl;
std::cout << "hash = " << bc->blocks[i]->_hash << std::endl << std::endl; } //退出之前 删除
delete bc; return ;
}

运行结果

下一个章节介绍工作量证明
本章节代码存储在QQ群文件中 文件名 MyBlockChainCppSample_part1

参考博文:

https://blog.csdn.net/simple_the_best/article/details/78073844

https://jeiwan.cc/posts/building-blockchain-in-go-part-1/

cpp 区块链模拟示例(三)新基本原形工程的建立的更多相关文章

  1. cpp 区块链模拟示例(一)工程建立

    /* 作 者: itdef 欢迎转帖 请保持文本完整并注明出处 技术博客 http://www.cnblogs.com/itdef/ 技术交流群 群号码:432336863欢迎c c++ window ...

  2. cpp 区块链模拟示例(四) 区块链工作量证明

    本文主要在之前的区块链原形上添加了工作量证明,并且为后继的交易功能做好准备. 上一个章节我们已经创建了区块链的基本原形,但是区块的哈希计算和加入太过于简单,如果按照这种速度添加区块那么区块链估计一个小 ...

  3. cpp 区块链模拟示例(五) 序列化

    有了区块和区块链的基本结构,有了工作量证明,我们已经可以开始挖矿了.剩下就是最核心的功能-交易,但是在开始实现交易这一重大功能之前,我们还要预先做一些铺垫,比如数据的序列化和启动命令解析. 根据< ...

  4. cpp 区块链模拟示例(七) 补充 Merkle树

    Merkle 树 完整的比特币数据库(也就是区块链)需要超过 140 Gb 的磁盘空间.因为比特币的去中心化特性,网络中的每个节点必须是独立,自给自足的,也就是每个节点必须存储一个区块链的完整副本.随 ...

  5. cpp 区块链模拟示例(二)工程代码解析

    /* 作 者: itdef 欢迎转帖 请保持文本完整并注明出处 技术博客 http://www.cnblogs.com/itdef/ 技术交流群 群号码:432336863欢迎c c++ window ...

  6. cpp 区块链模拟示例(六) 交易

    交易(transaction)是比特币的核心所在,而区块链的唯一目的,也正是为了能够安全可靠地存储交易.在区块链中,交易一旦被创建,就没有任何人能够再去修改或是删除它.在今天的文章中,我们会实现交易的 ...

  7. 使用Geth 构建以太坊区块链并模拟挖矿过程

    使用Geth 构建以太坊区块链并模拟挖矿过程 Go-ethereum 是以太坊官方的一个Golang 实现,我们可以使用Geth 工具来创建创世区块并启动区块链,使用Clef 实现以太坊钱包的功能,以 ...

  8. 一、Bitcoin比特币与BlockChain区块链技术

    一.比特币历史 2008 年 10 月 31 日,一个网名叫中本聪(英文翻译过来滴)的家伙发布比特币唯一的白皮书:<Bitcoin:A Peer-to-PeerElectronic Cash S ...

  9. 001-Bitcoin比特币与BlockChain区块链技术

    一.比特币历史 2008 年 10 月 31 日,一个网名叫中本聪(英文翻译过来滴)的家伙发布比特币唯一的白皮书:<Bitcoin:A Peer-to-PeerElectronic Cash S ...

随机推荐

  1. PAT 甲级 1027 Colors in Mars (20 分)

    1027 Colors in Mars (20 分) People in Mars represent the colors in their computers in a similar way a ...

  2. JVM学习总结(一):Java内存区域

    一.JVM运行时数据区 1.程序计数器: (1)一块较小的线程私有的内存空间. (2)JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器(或一个内核) ...

  3. C# 栈 、队列的概念

    栈: 也是System.Collections下的数据结构 存储依然是Object类型的对象 Stack 名字 = new Stack(); Count:实际拥有的元素个数 栈的释放顺序是先进后出(后 ...

  4. ubuntu 16.04 mysql5.7.17 开放远程3306端口

    ubuntu 16.04 mysql5.7.17 开放远程3306端口 原创 2017年01月19日 20:33:27 标签: mysql / ubuntu 2644 开启mysql的远程访问权限 默 ...

  5. MYSQL与TiDB的执行计划

    前言 这里采用了tpc-h一个数据库的数据量来进行查询计划的对比.并借助tpc-h中的22条查询语句进行执行计划分析. mysql采用的是标准安装,TiDB采用的是单机测试版,这里的性能结果不能说明其 ...

  6. __iter__ 和 __next__

    class F: def __init__(self,x): self.x = x def __iter__(self): #把对象 变成可迭代对象 return self def __next__( ...

  7. Access restriction 问题解决

    最近导入新项目,导入所有用到的jar包,发现其中一个引用报错 import com.sun.istack.internal.Nullable; 具体信息如下: Access restriction: ...

  8. MySQL InnoDB内存压力判断以及存在的疑问

    本文出处:http://www.cnblogs.com/wy123/p/7259866.html(保留出处并非什么原创作品权利,本人拙作还远远达不到,仅仅是为了链接到原文,因为后续对可能存在的一些错误 ...

  9. MySQL中基于mysqldump和二进制日志log-bin进行逻辑备份以及基于时间点的还原

    本文出处:http://www.cnblogs.com/wy123/p/6956464.html 本文仅模拟使用mysqldump和log-bin二进制日志进行简单测试,仅作为个人学习笔记,可能离实际 ...

  10. Mask RCNN 源码阅读(update)

    之前看了Google官网的object_dectect 的源码,感觉Google大神写的还不错.最近想玩下Mask RCNN,就看了下源码,这里刚好当做总结和梳理.链接如下: Google官网的obj ...