看完此文再不懂区块链算我输,用Python从零开始创建区块链

如果你还没有听说过 3 点钟区块链群,说明你还不是链圈的人;如果你还没有加入 3 点钟区块链群,说明你还不是链圈的大佬;如果你还没有被 3 点钟区块链群刷屏,说明你还体会不到什么是“币圈一天,人间一年”。

“三点钟区块链”无疑成为了大家春节期间焦虑的根源,而“区块链”注定是 2018 年被持续讨论、关注的行业性热点话题。

3 月 1 日,朱啸虎对正翻涌不断的区块链热潮再次开炮,在朋友圈一张画满区块链应用的图上,朱啸虎质疑:所有这些应用加在一起,有多少日活用户?“2000 年的互联网泡沫至少还有 eyeball,今天的区块链除了炒币外还有什么”?

在此之前,朱啸虎在朋友圈转发了讽刺区块链投资热的文章《来,喝了这碗区块链的毒鸡汤!》,并声明:“不要拉我进任何 3 点钟群,有些风口宁愿错过,有些钱宁愿不赚,大家晚节保重。”朱啸虎还表示,说 ICO 是庞氏骗局是在侮辱庞氏骗局。

作为程序员的你,再不懂这个技术,2018 可能会被淘汰!下面和小编一起从十个幽默段子入门区块链吧!

笑喷!区块链十个段子集锦

1、假如你是一位女性,你男朋友每次跟你说一句肉麻的话或者承诺给你买东西,你都立刻录下来并且发给你的和他的所有闺蜜、同学、同事,还有各种群和朋友圈,让他再也无法抵赖,这叫区块链。

2、麻将是中国传统的区块链项目:四个矿工一组,先碰撞出 13 个数字正确哈希值的矿工可以获得记账权并得到奖励。不可篡改。因为说服其他三个人需要消耗太多算力和体力。

3、玩夜店的小姐姐和玩虚拟币的小哥哥们有几处相似:

  • 都是自认聪明的优秀群体
  • 不给自己赚钱的都是傻 X 屌丝
  • 都认识很多大佬
  • 都明白很多道理
  • 都是在等自己涨价或者自己的虚拟币涨价被别人接盘

4、区块链是正经技术,各种币正不正经就不知道了。

5、吴三桂在山海关冲冠一怒,本质是为了争夺睡陈圆圆的权力;大佬们在区块链路上的互怼,本质是为了争夺割韭菜的权力。

6、新学期刚开始,儿子问老爸:「父亲工作一栏怎么填?是写币民吗?」老爸犹豫了一下说:「就写多家上市公司股东。」

7、最近数字货币很火,很多山寨币都是几倍几十倍的增长。很多炒币的开始飘飘然,叫嚣什么「一币一嫩模」。有朋友就问我要不要跟?我观点很简单:淘金热,一窝蜂淘金,风险很大。所以,让他们叫「一币一嫩模」去吧,不要跟风盲目炒币,我们应该赚他们的钱——去当嫩模!当嫩模!嫩模!

8、我昨天遇见一币友,问他:「近来币市暴降,睡觉质量怎么样?」

他说:「还行,像婴儿般睡觉!」

我说:「羡慕了。」

他说:「是睡一个小时,醒了,然后哭一个小时,接着,再睡一个小时,起来再哭一个小时。」

9、老同志语重心长地对 80、90 后说:「别玩那些比特币,那些虚拟的玩意,做点实事在北京买个房、娶个媳妇,多好!」90 后回答说:「你们都把几千块钱成本的房子搞到10万一平米了。我们不另寻出路,搞一串串数字 10 万一个卖给你们,我们拿什么买得起房子啊?」

10、首先感谢公司拿出价值 100 万的比特币作为给员工的奖励,其次我觉得自己很幸运能拿到这 95 万的奖励,然后我觉得我还是要好好规划一下这 86 万的用处,毕竟 70 万也不是一笔小钱,我打算拿出 20 万给父母,剩下的 36 万暂时还没想好怎么用,总之,感谢公司价值 30 万的比特币的奖励,谢谢,祝大家和我一样都能得到这 15 万的奖励。

图解:大白话了解到底区块链是个啥?

what is 区块链

“区块链仅仅是一门技术而已”,“比特币”仅仅是区块链技术的一种应用而已, 就好比,一个人会厨艺的技术,但是应用起他的厨艺可以做出“宫保鸡丁”、”鱼香肉丝”等各式的菜肴。

那么,到底“区块链”是个啥?我们这里借助网上一个比较流行的段子,将它用图形的形式展示给大家。

我们可以将“比特币”抽象成“某荣的照片”如上图所示,如果网上很多用户想要得到某荣的照片,需要到一个固定的网站去搜索。

当然你也没其他地方可去嘛,那么好了,这个某榴网站天天给你弹出广告啊,小窗口啊,你都得忍着,没办法,因为就这一个地方可以获取嘛。

再者,这个网站突然被警察叔叔封杀了咋办?或者断电?断网?管他啥的,反正就是服务器崩溃了,那么悲剧的“某荣”粉丝们心爱的 2100 张照片将全部失去提供资源的场所。

这就是网上流传的新词“中心化”喽,他的弊端就是资源集中一起,抗风险容错性很弱。资源容易丢失。

那么,怎么解决这个问题呢?我们试想,能不能让每个“某荣”粉丝,都拥有这 2100 张照片呢?比如下图:

这样的话,貌似就不用在依赖那个“某榴”网站了呢。即使某个粉丝突然电脑崩溃,他随便找一个其他粉丝来获得这 2100 张照片,不愁了,不愁了。

后来,一个叫“某本聪”的一个虚拟人物提供了,一个拥有协议的某荣照片共享文件夹,用户可以从中获取照片,但是必须遵循协议。

这样,每个粉丝都可以从这个文件夹中获取那 2100 张某荣的照片,但是全部获取的粉丝必须要遵循一个协议,当然是人家某本聪定义的协议啦。

“不得复制,修改,共享文件中的任意照片,粉丝们在共享文件夹中的任何行为都会被记录,并且是按照时间去记录!”

粉丝们这么喜爱某荣,而且不需要去某榴获取,当然就纷纷踊跃加入了~

忽然,有一天,调皮的小 One 想要违背规则,在 2018 年 1 月 15 日中午 12 点删除编号为 1-100 的某荣照片。

根据协议,这个行为会被记录,并且会广播给其他粉丝。

照片到底被删除了么?当然不是,因为小露手里也有照片嘛,她收到广播后可以立刻恢复共享文件夹中已经删除的照片,小 One 永远别想对“共享文件夹”搞修改破坏,且所有行为都同步记录在其他用户的电脑里。

这就是区块链,数据分散存储,去中心化。按时间戳广播记录所有行为,无法修改、破坏数据源或造假。

除非同一时刻炸掉 100 万个用户的电脑,或互联网消失,或世界毁灭....否则数据将永远存在~~

如何增加区块链保护的资源?“某本聪”又来了。他说,你们是可以在文件中添加某荣照片的,但是呢,你们各位必须达到某种“共识”?啥是“共识”,就是我们都承认的规则喽。

那到底是个啥共识呢?啊,小 One 和小露立刻了解了某本聪的意思,在每年规定的时间内,尽快拍出 100 张某荣的照片,这样就可以添加到“某荣共享文件夹”了,我们的资源就扩充了。

但是,好景不长。某本聪发现大量的拍照,冲击前 100,那很快就能达成了,拍照就没有难度了。

而且照片质量还差,好像谁都能轻易的添加照片资源,这样就保证不了某荣的照片质量了。于是,某本聪再次降临,增加拍照的共识难度。 

小 One 和小露作为忠实的粉丝,怎能放弃,他们买了高端相机,让某荣摆出各种姿势,花费大量的时间和汗水完成了高质量的照片,当然自己也非常的辛苦。

这样高质量的照片就可以添加到“某荣的文件夹”中了。

何为 ICO?

小 One 想:每张照片都是不可造假破坏的,所以具有唯一性,还有单独编号,给每一张照片估价,它不就值钱了吗?就像现实世界中无法复制的名画一样!

小 One 就把之前的某荣照片,构造出来相应的“某荣币”,当然这个行为就向我们的政府通过国库的黄金数额发行等价的人民币类似啦。

估值呢,当然小 One 说的算啦,人家是照片的拥有者嘛。

为了证实某荣币 5W 块一个,小 One 首先购买了 2100 个中的 1100 个,剩下的发行给吃瓜群众们啦,我们已经用 5W 买一个,说明他已经值这个价钱啦,剩下 1000 个大家一起买吧。

这样的话如果 2100 个全部认购出去,2100 个某荣币可以就估值 1.05 亿块哦~这种通过数字货币的发行而得到融资的过程就是 ICO 啦。

根据这个图的含义,如果小 One 和小露是一个可信任的机构或者公众人物,还是可以相信他们的,当然啦也会有很多不法分子恶意发行货币来套现的。

这也是我们国家为什么禁止 ICO 发行的原因啦,因为目前没有一个完善的 ICO 监管条例能够保证发行货币的机构的可信任性和合法的监管他们,所以吃瓜群众们要自己承担风险找到一个可信的机构。

这样 2100 个某荣币全部认购成功,基金成立,这就是 ICO。当然吃瓜群众也可以继续拍照创造某荣币,就是有点难罢了。

现在各位了解什么是区块链和 ICO 了吧~下面手把手教你如何用 Python 语言创建一个区块链?

用 Python 从 0 开始创建一个区块链

对数字货币的崛起感到新奇的我们,并且想知道其背后的技术——区块链是怎样实现的。本文通过 Python 构建一个区块链可以加深对区块链的理解。

准备工作

本文要求读者对 Python 有基本的理解,能读写基本的 Python,并且需要对 HTTP 请求有基本的了解。

我们知道区块链是由区块的记录构成的不可变、有序的链结构,记录可以是交易、文件或任何你想要的数据,重要的是它们是通过哈希值(hashes)链接起来的。

如果你还不是很了解哈希,可以查看这篇文章https://learncryptography.com/hash-functions/what-are-hash-functions。

环境准备:

环境准备,确保已经安装 Python3.6+、pip、Flask、requests。

安装方法:

  1. pip install Flask==0.12.2 requests==2.18.4 

同时还需要一个 HTTP 客户端,比如 Postman、cURL 或其他客户端。

参考源代码(原代码在我翻译的时候,无法运行,我 fork 了一份,修复了其中的错误,并添加了翻译,感谢 star)。

开始创建 Blockchain

新建一个文件 blockchain.py,本文所有的代码都写在这一个文件中,可以随时参考源代码。

Blockchain 类

首先创建一个 Blockchain 类,在构造函数中创建了两个列表,一个用于储存区块链,一个用于储存交易。

以下是 Blockchain 类的框架:

  1. class Blockchain(object):  
  2.     def __init__(self):  
  3.         self.chain = []  
  4.         self.current_transactions = []  
  5.     def new_block(self):  
  6.         # Creates a new Block and adds it to the chain  
  7.         pass  
  8.     def new_transaction(self):  
  9.         # Adds a new transaction to the list of transactions  
  10.         pass  
  11.     @staticmethod  
  12.     def hash(block):  
  13.         # Hashes a Block  
  14.         pass  
  15.     @property  
  16.     def last_block(self):  
  17.         # Returns the last Block in the chain  
  18.         pass 

Blockchain 类用来管理链条,它能存储交易、加入新块等,下面我们来进一步完善这些方法。

块结构

每个区块包含属性:索引(index)、Unix 时间戳(timestamp)、交易列表(transactions)、工作量证明(稍后解释)以及前一个区块的 Hash 值。

以下是一个区块的结构:

  1. block = {  
  2.     'index': 1,  
  3.     'timestamp': 1506057125.900785,  
  4.     'transactions': [  
  5.         {  
  6.             'sender': "8527147fe1f5426f9dd545de4b27ee00",  
  7.             'recipient': "a77f5cdfa2934df3954a5c7c7da5df1f",  
  8.             'amount': 5,  
  9.         }  
  10.     ],  
  11.     'proof': 324984774000,  
  12.     'previous_hash': "2cf24dba5fb0a30e26e83b2ac5b9e29e1b161e5c1fa7425e73043362938b9824"  

到这里区块链的概念就清楚了,每个新的区块都包含上一个区块的 Hash,这是关键的一点,它保障了区块链的不可变性。

如果攻击者破坏了前面的某个区块,那么后面所有区块的Hash都会变得不正确。不理解的话,慢慢消化,可参考{% post_link whatbc 区块链技术原理 %}。

加入交易

接下来我们需要添加一个交易,来完善下 new_transaction 方法:

  1. class Blockchain(object):  
  2.     ...  
  3.     def new_transaction(self, sender, recipient, amount):  
  4.         """  
  5.         生成新交易信息,信息将加入到下一个待挖的区块中  
  6.         :param sender: <str> Address of the Sender  
  7.         :param recipient: <str> Address of the Recipient  
  8.         :param amount: <int> Amount  
  9.         :return: <int> The index of the Block that will hold this transaction 
  10.         """  
  11.         self.current_transactions.append({  
  12.             'sender': sender,  
  13.             'recipient': recipient,  
  14.             'amount': amount,  
  15.         })  
  16.         return self.last_block['index'] + 1 

方法向列表中添加一个交易记录,并返回该记录将被添加到的区块(下一个待挖掘的区块)的索引,等下在用户提交交易时会有用。

创建新块

当 Blockchain 实例化后,我们需要构造一个创世块(没有前区块的第一个区块),并且给它加上一个工作量证明。每个区块都需要经过工作量证明,俗称挖矿,稍后会继续讲解。

为了构造创世块,我们还需要完善 new_block(),new_transaction() 和hash() 方法:

  1. import hashlib  
  2. import json  
  3. from time import time  
  4. class Blockchain(object):  
  5.     def __init__(self):  
  6.         self.current_transactions = []  
  7.         self.chain = []  
  8.         # Create the genesis block  
  9.         self.new_block(previous_hash=1, proof=100)  
  10.     def new_block(self, proof, previous_hash=None):  
  11.         """  
  12.         生成新块  
  13.         :param proof: <int> The proof given by the Proof of Work algorithm  
  14.         :param previous_hash: (Optional) <str> Hash of previous Block  
  15.         :return: <dict> New Block  
  16.         """  
  17.         block = {  
  18.             'index': len(self.chain) + 1,  
  19.             'timestamp': time(),  
  20.             'transactions': self.current_transactions,  
  21.             'proof': proof,  
  22.             'previous_hash': previous_hash or self.hash(self.chain[-1]),  
  23.         } 
  24.  
  25.         # Reset the current list of transactions  
  26.         self.current_transactions = []  
  27.         self.chain.append(block)  
  28.         return block 
  29.  
  30.     def new_transaction(self, sender, recipient, amount):  
  31.         """  
  32.         生成新交易信息,信息将加入到下一个待挖的区块中  
  33.         :param sender: <str> Address of the Sender  
  34.         :param recipient: <str> Address of the Recipient  
  35.         :param amount: <int> Amount  
  36.         :return: <int> The index of the Block that will hold this transaction  
  37.         """  
  38.         self.current_transactions.append({  
  39.             'sender': sender,  
  40.             'recipient': recipient,  
  41.             'amount': amount,  
  42.         })  
  43.         return self.last_block['index'] + 1  
  44.     @property  
  45.     def last_block(self):  
  46.         return self.chain[-1]  
  47.     @staticmethod  
  48.     def hash(block):  
  49.         """  
  50.         生成块的 SHA-256 hash值  
  51.         :param block: <dict> Block  
  52.         :return: <str>  
  53.         """  
  54.         # We must make sure that the Dictionary is Ordered, or we'll have inconsistent hashes  
  55.         block_string = json.dumps(block, sort_keys=True).encode()  
  56.         return hashlib.sha256(block_string).hexdigest() 

通过上面的代码和注释可以对区块链有直观的了解,接下来我们看看区块是怎么挖出来的。

理解工作量证明

新的区块依赖工作量证明算法(PoW)来构造。PoW 的目标是找出一个符合特定条件的数字,这个数字很难计算出来,但容易验证。这就是工作量证明的核心思想。

为了方便理解,举个例子:

假设一个整数 x 乘以另一个整数 y 的积的 Hash 值必须以 0 结尾,即 hash(x * y) = ac23dc...0。设变量 x = 5,求 y 的值?

用 Python 实现如下:

  1. from hashlib import sha256  
  2. x = 5  
  3. y = 0  # y未知  
  4. while sha256(f'{x*y}'.encode()).hexdigest()[-1] != "0":  
  5.     y += 1  
  6. print(f'The solution is y = {y}') 

结果是:y = 21,因为:

  1. hash(5 * 21) = 1253e9373e...5e3600155e860 

在比特币中,使用称为 Hashcash 的工作量证明算法,它和上面的问题很类似,矿工们为了争夺创建区块的权利而争相计算结果。

通常,计算难度与目标字符串需要满足的特定字符的数量成正比,矿工算出结果后,会获得比特币奖励。当然,在网络上非常容易验证这个结果。

实现工作量证明

让我们来实现一个相似 PoW 算法,规则是:寻找一个数 p,使得它与前一个区块的 proof 拼接成的字符串的 Hash 值以 4 个零开头。

  1. import hashlib  
  2. import json  
  3. from time import time  
  4. from uuid import uuid4  
  5. class Blockchain(object):  
  6.     ...  
  7.     def proof_of_work(self, last_proof):  
  8.         """  
  9.         简单的工作量证明:  
  10.          - 查找一个 p' 使得 hash(pp') 以4个0开头  
  11.          - p 是上一个块的证明,  p' 是当前的证明  
  12.         :param last_proof: <int>  
  13.         :return: <int>  
  14.         """  
  15.         proof = 0  
  16.         while self.valid_proof(last_proof, proof) is False:  
  17.             proof += 1  
  18.         return proof  
  19.     @staticmethod  
  20.     def valid_proof(last_proof, proof):  
  21.         """  
  22.         验证证明: 是否hash(last_proof, proof)以4个0开头?  
  23.         :param last_proof: <int> Previous Proof  
  24.         :param proof: <int> Current Proof  
  25.         :return: <bool> True if correct, False if not.  
  26.         """  
  27.         guess = f'{last_proof}{proof}'.encode()  
  28.         guess_hash = hashlib.sha256(guess).hexdigest()  
  29.         return guess_hash[:4] == "0000" 

衡量算法复杂度的办法是修改零开头的个数。使用 4 个零来用于演示,你会发现多一个零都会大大增加计算出结果所需的时间。

现在 Blockchain 类基本已经完成了,接下来使用 HTTP requests 来进行交互。

Blockchain 作为 API 接口

我们将使用 Python Flask 框架,这是一个轻量 Web 应用框架,它方便将网络请求映射到 Python 函数,现在我们来让 Blockchain 运行在 Flask Web 上。

我们将创建三个接口:

  • /transactions/new 创建一个交易并添加到区块
  • /mine 告诉服务器去挖掘新的区块
  • /chain 返回整个区块链

创建节点

我们的“Flask 服务器”将扮演区块链网络中的一个节点,我们先添加一些框架代码:

  1. import hashlib  
  2. import json  
  3. from textwrap import dedent  
  4. from time import time  
  5. from uuid import uuid4  
  6. from flask import Flask  
  7. class Blockchain(object):  
  8.     ...  
  9. # Instantiate our Node  
  10. app = Flask(__name__)  
  11. # Generate a globally unique address for this node  
  12. node_identifier = str(uuid4()).replace('-', '')  
  13. # Instantiate the Blockchain  
  14. blockchain = Blockchain()  
  15. @app.route('/mine', methods=['GET'])  
  16. def mine():  
  17.     return "We'll mine a new Block"  
  18. @app.route('/transactions/new', methods=['POST'])  
  19. def new_transaction():  
  20.     return "We'll add a new transaction"  
  21. @app.route('/chain', methods=['GET'])  
  22. def full_chain():  
  23.     response = {  
  24.         'chain': blockchain.chain,  
  25.         'length': len(blockchain.chain),  
  26.     }  
  27.     return jsonify(response), 200  
  28. if __name__ == '__main__':  
  29.     app.run(host='0.0.0.0', port=5000) 

简单的说明一下以上代码:

  • 第 15 行:创建一个节点。
  • 第 18 行:为节点创建一个随机的名字。
  • 第 21 行:实例 Blockchain 类。
  • 第 24–26 行:创建 /mine GET 接口。
  • 第 28–30 行:创建 /transactions/new POST 接口,可以给接口发送交易数据。
  • 第 32–38 行:创建 /chain 接口, 返回整个区块链。
  • 第 40–41 行:服务运行在端口 5000 上。

发送交易

发送到节点的交易数据结构如下:

  1. {  
  2.  "sender": "my address",  
  3.  "recipient": "someone else's address",  
  4.  "amount": 5  

之前已经有添加交易的方法,基于接口来添加交易就很简单了:

  1. import hashlib  
  2. import json  
  3. from textwrap import dedent  
  4. from time import time  
  5. from uuid import uuid4  
  6. from flask import Flask, jsonify, request  
  7. ...  
  8. @app.route('/transactions/new', methods=['POST'])  
  9. def new_transaction():  
  10.     values = request.get_json()  
  11.     # Check that the required fields are in the POST'ed data  
  12.     required = ['sender', 'recipient', 'amount']  
  13.     if not all(k in values for k in required): 
  14.         return 'Missing values', 400  
  15.     # Create a new Transaction  
  16.     index = blockchain.new_transaction(values['sender'], values['recipient'], values['amount'])  
  17.     response = {'message': f'Transaction will be added to Block {index}'}  
  18.     return jsonify(response), 201 

挖矿

挖矿正是神奇所在,它很简单,做了以下三件事:

  • 计算工作量证明 PoW。
  • 通过新增一个交易授予矿工(自己)一个币。
  • 构造新区块并将其添加到链中。 
  1. import hashlib  
  2. import json  
  3. from textwrap import dedent  
  4. from time import time  
  5. from uuid import uuid4  
  6. from flask import Flask, jsonify, request  
  7. ...  
  8. import hashlib  
  9. import json  
  10. from time import time  
  11. from uuid import uuid4  
  12. from flask import Flask, jsonify, request  
  13. ...  
  14. @app.route('/mine', methods=['GET'])  
  15. def mine():  
  16.     # We run the proof of work algorithm to get the next proof...  
  17.     last_block = blockchain.last_block  
  18.     last_proof = last_block['proof']  
  19.     proof = blockchain.proof_of_work(last_proof)  
  20.     # 给工作量证明的节点提供奖励. 
  21.     # 发送者为 "0" 表明是新挖出的币  
  22.     blockchain.new_transaction(  
  23.         sender="0",  
  24.         recipient=node_identifier,  
  25.         amount=1,  
  26.     )  
  27.     # Forge the new Block by adding it to the chain  
  28.     block = blockchain.new_block(proof)  
  29.     response = {  
  30.         'message': "New Block Forged", 
  31.         'index': block['index'],  
  32.         'transactions': block['transactions'],  
  33.         'proof': block['proof'],  
  34.         'previous_hash': block['previous_hash'],  
  35.     }  
  36.     return jsonify(response), 200 

注意交易的接收者是我们自己的服务器节点,我们做的大部分工作都只是围绕 Blockchain 类方法进行交互。到此,我们的区块链就算完成了,我们来实际运行下。

运行区块链

你可以使用 cURL 或 Postman 去和 API 进行交互。

启动 server:

  1. $ python blockchain.py  
  2. * Runing on http://127.0.0.1:5000/ (Press CTRL+C to quit) 

让我们通过请求 http://localhost:5000/mine 来进行挖矿:

通过 post 请求,添加一个新交易:

如果不是使用 Postman,则用以下的 cURL 语句也是一样的:

  1. $ curl -X POST -H "Content-Type: application/json" -d '{  
  2.  "sender": "d4ee26eee15148ee92c6cd394edd974e",  
  3.  "recipient": "someone-other-address",  
  4.  "amount": 5  
  5. }' "http://localhost:5000/transactions/new" 

在挖了两次矿之后,就有 3 个块了,通过请求 http://localhost:5000/chain 可以得到所有的块信息。

  1. {  
  2.   "chain": [  
  3.     {  
  4.       "index": 1,  
  5.       "previous_hash": 1,  
  6.       "proof": 100,  
  7.       "timestamp": 1506280650.770839,  
  8.       "transactions": []  
  9.     },  
  10.     {  
  11.       "index": 2,  
  12.       "previous_hash": "c099bc...bfb7",  
  13.       "proof": 35293,  
  14.       "timestamp": 1506280664.717925,  
  15.       "transactions": [  
  16.         {  
  17.           "amount": 1,  
  18.           "recipient": "8bbcb347e0634905b0cac7955bae152b",  
  19.           "sender": "0"  
  20.         }  
  21.       ]  
  22.     },  
  23.     {  
  24.       "index": 3,  
  25.       "previous_hash": "eff91a...10f2",  
  26.       "proof": 35089,  
  27.       "timestamp": 1506280666.1086972,  
  28.       "transactions": [  
  29.         {  
  30.           "amount": 1,  
  31.           "recipient": "8bbcb347e0634905b0cac7955bae152b",  
  32.           "sender": "0"  
  33.         }  
  34.       ]  
  35.     }  
  36.   ],  
  37.   "length": 3  

一致性(共识)

我们已经有了一个基本的区块链可以接受交易和挖矿,但是区块链系统应该是分布式的。

既然是分布式的,那么我们究竟拿什么保证所有节点有同样的链呢?这就是一致性问题,我们要想在网络上有多个节点,就必须实现一个一致性的算法。

注册节点

在实现一致性算法之前,我们需要找到一种方式让一个节点知道它相邻的节点。

每个节点都需要保存一份包含网络中其他节点的记录,因此让我们新增几个接口:

  • /nodes/register 接收 URL 形式的新节点列表。
  • /nodes/resolve 执行一致性算法,解决任何冲突,确保节点拥有正确的链。

我们修改下 Blockchain 的 init 函数并提供一个注册节点方法:

  1. ...  
  2. from urllib.parse import urlparse  
  3. ...  
  4. class Blockchain(object):  
  5.     def __init__(self):  
  6.         ...  
  7.         self.nodes = set()  
  8.         ...  
  9.     def register_node(self, address):  
  10.         """  
  11.         Add a new node to the list of nodes  
  12.         :param address: <str> Address of node. Eg. 'http://192.168.0.5:5000'  
  13.         :return: None  
  14.         """  
  15.         parsed_url = urlparse(address)  
  16.         self.nodes.add(parsed_url.netloc) 

我们用 set 来储存节点,这是一种避免重复添加节点的简单方法。

实现共识算法

前面提到,冲突是指不同的节点拥有不同的链,为了解决这个问题,规定最长的、有效的链才是最终的链,换句话说,网络中有效最长链才是实际的链。

我们使用以下的算法,来达到网络中的共识:

  1. ...  
  2. import requests  
  3. class Blockchain(object)  
  4.     ...  
  5.     def valid_chain(self, chain):  
  6.         """  
  7.         Determine if a given blockchain is valid  
  8.         :param chain: <list> A blockchain  
  9.         :return: <bool> True if valid, False if not  
  10.         """  
  11.         last_block = chain[0]  
  12.         current_index = 1  
  13.         while current_index < len(chain):  
  14.             block = chain[current_index]  
  15.             print(f'{last_block}')  
  16.             print(f'{block}')  
  17.             print("\n-----------\n")  
  18.             # Check that the hash of the block is correct  
  19.             if block['previous_hash'] != self.hash(last_block):  
  20.                 return False  
  21.             # Check that the Proof of Work is correct  
  22.             if not self.valid_proof(last_block['proof'], block['proof']):  
  23.                 return False  
  24.             last_block = block  
  25.             current_index += 1  
  26.         return True  
  27.     def resolve_conflicts(self): 
  28.         """  
  29.         共识算法解决冲突  
  30.         使用网络中最长的链.  
  31.         :return: <bool> True 如果链被取代, 否则为False  
  32.         """  
  33.         neighbours = self.nodes  
  34.         new_chain = None  
  35.         # We're only looking for chains longer than ours  
  36.         max_length = len(self.chain)  
  37.         # Grab and verify the chains from all the nodes in our network  
  38.         for node in neighbours:  
  39.             response = requests.get(f'http://{node}/chain')  
  40.             if response.status_code == 200:  
  41.                 length = response.json()['length'] 
  42.                 chain = response.json()['chain']  
  43.                 # Check if the length is longer and the chain is valid  
  44.                 if length > max_length and self.valid_chain(chain):  
  45.                     max_length = length  
  46.                     new_chain = chain  
  47.         # Replace our chain if we discovered a new, valid chain longer than ours  
  48.         if new_chain:  
  49.             self.chain = new_chain  
  50.             return True  
  51.         return False 

第一个方法 valid_chain() 用来检查是否是有效链,遍历每个块验证 hash 和 proof。

第二个方法 resolve_conflicts() 用来解决冲突,遍历所有的邻居节点,并用上一个方法检查链的有效性, 如果发现有效更长链,就替换掉自己的链。

让我们添加两个路由,一个用来注册节点,一个用来解决冲突。

  1. @app.route('/nodes/register', methods=['POST'])  
  2. def register_nodes():  
  3.     values = request.get_json()  
  4.     nodes = values.get('nodes')  
  5.     if nodes is None:  
  6.         return "Error: Please supply a valid list of nodes", 400  
  7.     for node in nodes:  
  8.         blockchain.register_node(node)  
  9.     response = {  
  10.         'message': 'New nodes have been added',  
  11.         'total_nodes': list(blockchain.nodes),  
  12.     }  
  13.     return jsonify(response), 201  
  14. @app.route('/nodes/resolve', methods=['GET'])  
  15. def consensus():  
  16.     replaced = blockchain.resolve_conflicts()  
  17.     if replaced:  
  18.         response = {  
  19.             'message': 'Our chain was replaced',  
  20.             'new_chain': blockchain.chain  
  21.         }  
  22.     else:  
  23.         response = {  
  24.             'message': 'Our chain is authoritative',  
  25.             'chain': blockchain.chain  
  26.         }  
  27.     return jsonify(response), 200 

你可以在不同的机器运行节点,或在一台机机开启不同的网络端口来模拟多节点的网络。

这里在同一台机器开启不同的端口演示,在不同的终端运行以下命令,就启动了两个节点:

  • http://localhost:5000 
  • http://localhost:5001 
  1. pipenv run python blockchain.py 
  2.  
  3. pipenv run python blockchain.py -p 5001 

然后在节点 2 上挖两个块,确保是更长的链,然后在节点 1 上访问接口 /nodes/resolve,这时节点 1 的链会通过共识算法被节点 2 的链取代。

好啦,你可以邀请朋友们一起来测试你的区块链。

编辑:陶家龙、孙淑娟

来源:互联网综合整理

  • 《笑喷!区块链十个段子集锦》转载自刘兴亮时间微信公众号

  • 《图解:大白话了解到底区块链是个啥?》转载自黑马程序员视频库微信公众号

  • 《用Python从零开始创建区块链》作者:Daniel van Flymen,Tiny熊译。

    原文:https://learnblockchain.cn/2017/10/27/build_blockchain_by_python/

    源码:https://github.com/xilibi2003/blockchain

转载于:https://www.cnblogs.com/chendongsheng/p/8537496.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/267478.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

重新加一个window_Activity、View、Window关系,进程间通信,责任链模式,Https,数据存储...

码仔&#xff0c;今天就给大家带来了《每日一道面试题》的第九期&#xff1a;01理解Activity View window的关系 Activity像一个工匠(控制单元)&#xff0c;Window像窗户(承载模型)&#xff0c;View像窗花(显示视图)LayoutInflater像剪刀&#xff0c;Xml配置像窗花图纸。 Activ…

排他网关(ExclusiveGateWay)

网关&#xff08;ExclusiveGateWay&#xff09; 作者&#xff1a;邓家海2018年3月11日 00:13:25 情景:某一家公司最近在给一个单位做一个财务审批的OA。具体需求是这样的&#xff1a;当部门申请的金额小于一万块的时候&#xff0c;财务部可以直接决策。当部分申请的金额大于一万…

离线安装宝塔lnmp_宝塔LNMP环境 Nginx安装EduSoho教程说明

[toc]宝塔下使用LNMP Nginx安装EduSoho创建站点 宝塔后台 > 网站 > 添加站点 > 输入信息 > 提交填写信息创建完成设置运行目录 宝塔后台 > 网站 > 管理运行目录选择web目录后保存 网站目录 > 运行目录 > 保存修改配置文件 配置文件 > 修改参数 >…

NHibernate初学者指南(10):一级和二级缓存

一级缓存 为了获得更好的性能&#xff0c;NHibernate智能地缓存数据。NHibernate有不同的缓存机制起作用&#xff0c;最重要的就是一级缓存。每个session对象维持一个一级缓存&#xff0c;session对象创建时缓存创建&#xff0c;session对象释放时缓存销毁。 缓存只不过是一个哈…

Freemarker模板引擎

模板引擎的实质就是将页面结构提前写好&#xff0c;然后将数据渲染到模板上生成一个静态页面&#xff0c;这样一来&#xff0c;下次就可以 直接访问静态文件&#xff0c;不用进行额外的获取数据的操作&#xff08;例如&#xff1a;访问数据库&#xff09;&#xff0c;这样大大提…

postgresql主从备份_基于windows平台的postgresql主从数据库流备份配置

基于windows平台的postgresql主从数据库流备份配置因工作需要&#xff0c;需要搞pg数据库的主从备份&#xff0c;领导给了个方向使用流备份&#xff0c;于是开始朝着这个方向进发。鸣谢大佬A_ccelerator的博客一、配置主从库1.环境准备对于 pg 的主从库配置&#xff0c;建议是使…

msvcrt.lib和LIBCD.lib链接冲突

今天在移植一个开源代码到windows的VC6工程&#xff0c;编译时出现了这些奇怪的LINK错误。 msvcrt.lib(MSVCRT.dll) : error LNK2005: _toupper already defined in LIBCD.lib(toupper.obj)msvcrt.lib(MSVCRT.dll) : error LNK2005: _tolower already defined in LIBCD.lib(to…

jq获取最后一个子节点_如何选择jQuery中的最后一个子元素?

牧羊人nacy如果要选择最后一个子元素&#xff0c;并且需要具体说明元素类型&#xff0c;则可以使用选择器last-of-type这是一个例子&#xff1a;$("div p:last-of-type").css("border", "3px solid red");$("div span:last-of-type").…

面向对象 - 继承/组合 - 总结

面向对象 - 继承:1.继承: 类与类之间的关系 什么是什么的关系 eg:人是动物 狗是动物 功能: 解决代码重用问题, 创建新类的方式, 类: 可继承一个或多个父类: 父类 基类/超类 类 派生类/子类 类: 对象之间相似的特征 父类:…

巧妙的有css合并图片解决tab切换的背景图片

巧妙的有css合并图片解决tab切换的背景图片 有时候做tab切换的时候 会碰到下面的这种情况 我截个图过来看看 tab切换 打开页面时候 茶庄介绍 及鼠标移上去时候 是上面这样的效果 当鼠标移下来的时候 是下面这样的 茶庄介绍 就变成这样的背景 一刚开始做这样的 我就想到用j…

XUPT_STA2018(部分题解)

A - 一方通行和最大公约数I CodeForces - 664A 作为学园都市最强的lv5&#xff0c;一方通行必须解决一道数学题才能接触last order身上植入的病毒&#xff0c;请你帮他解决这个问题。给出两个整数a,b 求出[a,b]区间中所有整数的最大公约数。输入输入包括一行&#xff0c;一…

java mongodb drivers 2升级到3_JAVA从头开始一基础梳理(3-2)

本章为大家介绍类的特性。首先&#xff0c;第一个特性----封装。在这个类别中&#xff0c;属性id 与属性 color 是可以被外部直接访问和修改的&#xff0c;比如可以看到&#xff0c;事实上这样的内容是错误的&#xff0c;我们需要在定义属性值时需要加入验证等等操作&#xff0…

C 语言第6节课

我上课C语言的第6节的笔记 &#xff1a;C语言真的需要多做多练多理解&#xff0c;不然真的学不懂.记不住呀。第一题&#xff1a;睇图片解题。main(){int x,y;scanf("%d",&x);if(x<0)printf("y%d",0);if(x>0&&x<5)printf("y%d&qu…

activiti idea 请假流程_IDEA开发流程Activiti需要注意的一些坑

1、慎用IDEA2020最新版本IDEA2020最新版本迎来了重大升级&#xff0c;对java和spring有了更好更强大的开发支持&#xff0c;但是对于Activiti流程开发却不再支持&#xff0c;可能是因为actiBPM插件太老&#xff0c;或者IDEA2020对插件支持做了规范要求等等&#xff0c;请各位喜…

VIPCA无法运行

本文转自ITPUB上一兄弟总结&#xff0c;以备查询。 今天一早来&#xff0c;同事说他周末在安装 rac 出问题了&#xff0c;始终无法使用vipca 创建 vip资源 &#xff0c; 叫我帮忙看看。 他用的是 vmware server 2.0 搭建 虚拟 rac 。 一. 环境如下  OS: Red Hat Enterprise …

python加权最小二乘_【Python】统计科学之加权最小二乘法

首页专栏python文章详情0统计科学之加权最小二乘法张俊红发布于 今天 10:03今天这篇来讲讲加权最小二乘法(WLS)&#xff0c;加权最小二乘是在普通的最小二乘回归(OLS)的基础上进行改造的&#xff0c;主要是用来解决异方差问题的。OLS的常规形式如下&#xff1a;我们在前面讲过O…

删除电脑中的mysql数据库吗_【数据库】怎么彻底删除mysql服务?

彻底删除mysql服务的步骤&#xff1a;1、关闭mysql服务在cmd命令行输入以下命令net stop mysql或者 我的电脑右键->管理->服务&#xff0c;进入后手动关闭。2、删除MySQL服务在cmd中&#xff0c;输入sc delete mysql&#xff0c;删除服务。但是MYSQL服务只是显示禁用状态…

mysql主从配置 简书_Mysql主从配置,实现读写分离-Go语言中文社区

转载&#xff1a;https://www.cnblogs.com/alvin_xp/p/4162249.html大型网站为了软解大量的并发访问&#xff0c;除了在网站实现分布式负载均衡&#xff0c;远远不够。到了数据业务层、数据访问层&#xff0c;如果还是传统的数据结构&#xff0c;或者只是单单靠一台服务器扛&am…

continue 的用户及实例

continue 跳出本次循环&#xff0c;继续下一次循环 “break”是跳出整个循环 实例1&#xff1a; for i in range(10): if i <5: continue    #这里条件是小于5&#xff0c;满足条件的就跳出了本次循环&#xff0c;不在执行print&#xff0c;所以最终打印的是5…

mysql报错注入实战_MySQL手工注入实战

实战记录&#xff0c;日本某站注入点 and 语句测试and11 返回正常&#xff0c;and2跳回首页&#xff0c;可能过滤了用 ’ 测试返回错误页面判断为注入点order by语句查询字段数测试字段数为9and 12 UNION SELECT 1,2,3,4,5,6,7,8,9报错联合查询语句&#xff0c;查询显示位2、3为…