This commit is contained in:
louzefeng
2024-07-09 18:38:56 +00:00
parent 8bafaef34d
commit bf99793fd0
6071 changed files with 1017944 additions and 0 deletions

View File

@@ -0,0 +1,166 @@
<audio id="audio" title="64 | 区块链技术细节:哈希算法" controls="" preload="none"><source id="mp3" src="https://static001.geekbang.org/resource/audio/db/0e/dbc4040c7424890b9e9e0940bef4a70e.mp3"></audio>
对于计算机来说,区块链就像一个单向链表,一个数据块中保存着三个信息。
- 真正的数据。
- 自己的地址或是ID
- 前一个数据块的地址。
这样通过追溯前一个块的地址把所有的数据块存成了一条链。所以我们叫其BlockChain。如下图所示。
<img src="https://static001.geekbang.org/resource/image/1c/62/1c75fed53ee023c353a31616d2b29e62.png" alt="" />
每个数据块的“地址”的编码使用了计算机上的一个算法计算机圈内人士把这个算法叫Secure Hash。有人音译为“安全哈希”也有人意译为“安全散列”。在计算机应用中hash算法主要有几个功能。
<li>
用来生成唯一标识一个数据块的ID身份证这个ID几乎不能重复。
</li>
<li>
用来做数据的特征码。只要数据中一个bit的数据出现更改那么整个hash值就完全不一样了。而且数学上保证了我们无法通过hash值反推回原数据。
</li>
于是很多公司在互联网上发布信息或软件的时候都会带上一个Checksum校验码。你只要把整个文件的数据传入到那个特定的hash算法中就会得到一串很长的字符串。如果和官方发布的Checksum字符串不一样那么就说明信息或文件内容被人更改或是信息残缺了。因此也被应用在“数字签名”中。
在计算机世界里有两个很著名的hash算法一个叫MD5[Wikipedia链接](https://en.wikipedia.org/wiki/MD5)一个叫SHA-2[Wikipedia链接](https://en.wikipedia.org/wiki/SHA-2) 区块链用的是SHA-256这个算法。
下面是一个示例。
<li>
&quot;chen hao&quot;这个信息计算MD5值得到 9824df83b2d35172ef5eb63a47a878eb一个16进制数
</li>
<li>
如果对 “chen hao&quot;做一个字符的修改,将字母&quot;o&quot;改成数字&quot;0”即 “chen ha0”计算出来的MD5值就成了 d521ce0616359df7e16b20486b78d2a8。可以看到这和之前的MD5值完全不一样了。
</li>
于是我们就可以利用hash算法的这个特性来对数据做&quot;数字签名&quot;。也就是说,我将&quot;数据&quot;和其&quot;签名&quot;hash计算值一起发布这样可以让收到方来验证数据有没有被修改。
我们再来看上面那个区块链的图。
<img src="https://static001.geekbang.org/resource/image/1c/62/1c75fed53ee023c353a31616d2b29e62.png" alt="" />
对于第一块数据我们把其“数据集”和“前数据块的hash值 00000a6cba”一起做hash值得到本区块的地址000007cabfa。然后下一个区块会把自己的数据和000007cabfa一起做hash得到000008acbed这个哈希值……如此往复下去。
根据“被hash的数据中有一个bit被修改了整个hash就完全不一样了”这个特性我们知道
<li>
如果前置数据块中的数据改了那么其hash就会完全不一样了也就是说你的ID或地址就变了于是别人就找不到这个数据块了
</li>
<li>
所以你还要去修改别人数据块中指向你的地址但是别人数据块中指向你的地址ID/hash变了也会导致他自己的地址ID/hash随之变化。因为他用你的地址生成了自己的地址这样一来你就需要把其他人的地址全部改掉。
</li>
在这样的连锁反应下你想要偷偷修改一个bit的难度一下就提高很多。所以在区块链的世界里越老的区块越安全也越不容易被人篡改越新的区块越不安全也越容易被人篡改。
# 比特币的hash算法
下面我来简单介绍一下,比特币中区块链的一些细节。下图是区块链的协议格式。
<img src="https://static001.geekbang.org/resource/image/a9/98/a99ad9de4d156cea9e8ee716c48e0298.png" alt="" />
其中VersionPrevious Block HashMerkle RootTimestampDifficulty Target 和Nonce这六个数据字段是区块链的区块数据协议头。后面的数据是交易数据分别是本块中的交易笔数H和交易列表最多不能超过1MB为什么是1MB后面会说
下面我来说一下区块头中的那六个字段的含义。
<li>
Version当前区块链协议的版本号4个字节。如果升级了这个版本号会变。
</li>
<li>
Previous Block Hash前面那个区块的hash地址。32个字节。
</li>
<li>
Merkle Root这个字段可以简单理解为是后面交易信息的hash值后面具体说明一下 。32个字节。
</li>
<li>
Timestamp区块生成的时间。这个时间不能早于前面11个区块的中位时间不能晚于&quot;网络协调时间&quot;——你所连接的所有结点时间的中位数。4个字节。
</li>
<li>
Bits也就是上图中的Difficulty Tagrget表明了当前的hash生成的难度后面会说。4个字节。
</li>
<li>
Nonce一个随机值用于找到满足某个条件的hash值。4字节。
</li>
对这六字段进行hash计算就可以得到本区块的hash值也就是其ID或是地址。其hash方式如下对区块头做两次SHA-256的hash求值
```
SHA-256(SHA-256 (Block Header))
```
当然事情并没有这么简单。比特币对这个hash值是有要求的其要求是那个Bits字段控制的然后你可以调整Nonce这个32位整型的值来找到符合条件的hash值。我们把这个事情叫做“挖矿”在下一篇中我们会详细讲一下这个事
# 关于 Merkle Root
前面说到过可以简单地将Merkle Root理解为交易的hash值。这里我们具体说一下比特币的Merkle Root是怎么计算出来的。
首先我们知道比特币的每一笔交易会有三个字段一个是转出方一个是转入方还有一个是金额。那么我们会对每个交易的这三个字段求hash然后把交易的hash做两两合并再求其hash直到算出最后一个hash值这就是我们的Merkle Root。
我画了一个图展示一下这个过程。
<img src="https://static001.geekbang.org/resource/image/d1/33/d196bfab2e786d055fa321055b7fef33.png" alt="" />
上面的示意图中有四笔交易A和B的hash成了Hash-AB C和D的hash成了Hash-CD然后再做Hash-AB + Hash-CD 的hash得到了Hash-ABCD这就是Merkle Root。整个过程就像一个二叉树一样。
下图是一个区块链的示意图,来自[比特币的白皮书](https://bitcoin.org/bitcoin.pdf)。
<img src="https://static001.geekbang.org/resource/image/62/5b/627b2acd3eef17785c9c7efcaf594a5b.png" alt="" />
为什么要这样做呢为什么不是把所有的交易都放在一起做一次hash呢这不也可以让人无法篡改吗这样做的好处是——我们把交易数据分成了若干个组。就像上面那个二叉树所表示的一样我们可以不断地把这个树分成左边的分支和右边的分支因为它们都被计算过hash值所以可以很快地校验其中的内容有没有被修改过。
这至少带来三个好处。
<li>
大量的交易数据可以被分成各种尺寸的小组,这样有利于我们整合数据和校验数据。
</li>
<li>
这样的开销在存储和内存上并不大,然而我们可以提高校验一组数据的难易程度。
</li>
<li>
在P2P的无中心化网络上我们可以把大量数据拆成一个一个小数据片传输可以提高网络的传输速度。
</li>
最后需要说一下的是以太坊有三个不同的Merkle Root树。因为以太坊要玩智能合约所以需要更多的Merkle Root。
<li>
一个是用来做交易hash的Merkle Root。
</li>
<li>
一个是用来表示状态State的。因为一个智能合同从初始状态走到最终状态需要有若干步也就是若干笔交易每一步都会让合同的状态发生变化所以需要保存合同的状态。
</li>
<li>
还有一个是用来做交易收据的。主要是用来记录一个智能合约中最终发生的交易信息。在StackExchange上的问题&quot;[Relationship between Transaction Trie and Receipts Trie](https://ethereum.stackexchange.com/questions/5888/relationship-between-transaction-trie-and-receipts-trie)&quot;中有相应的说明,你可以前往一看。
</li>
以太坊称其为Merkle Patricia Tree具体细节可参看其[官方的Wiki](https://github.com/ethereum/wiki/wiki/Patricia-Tree))。
# 比特币的交易模型
比特币区块中的交易数据其实也是一个链。为了讲清楚这个链我们需要了解一下比特币交易中的两个术语一个是input一个是output也就是交易的支出方input和收入方output
在比特币中一个交易可以有多个output也就是说我可以把一笔钱汇给多个人但一个output只能对应一个源的input还有一个条件就是output跟input的总数要吻合。
这里举个例子。假设Fred给了Alice 2个比特币Ted给了Alice 3个比特币这个时候Alice有5个比特币。然而大比特币的世界里是没有余额的所以对于Alice来说她只有两个没有花出去的交易一个是2个比特币一个是3个比特币。这在比特币中叫UTXOUnspent Transaction Output
此时如果Alice想要转给Bob 4个比特币她发现自己的两个交易中都不够也不能拆开之前的那两个比特币交易那么她只能把交易2和交易3当成input然后把自己和Bob当成outputBob分得4个她自己分1个。这样的交易才平衡。
<img src="https://static001.geekbang.org/resource/image/6c/41/6c82bd5c0bf3535b49f3dbe271480341.png" alt="" />
于是一笔交易可能会包含大量的Input和Output。因为比特币没有“余额”的概念所以需要通过多个input来凑然后output这边还需要给自己找零给矿工小费。
这样一来在比特币交易中你把钱给了我我又给了张三张三给了李四……就这样传递下去形成了一个交易链。因为还没有花出去所以就成了UTXO而系统计算你有没有钱可以汇出去时只需要查看一下你的UTXO就可以了。
<img src="https://static001.geekbang.org/resource/image/df/4d/dfd5d623ff27a5e6cfb0e5371822354d.png" alt="" /><br />
(图片来源:[https://bitcoin.org/en/developer-guide](https://bitcoin.org/en/developer-guide) )
UTXO因为没有账户和余额的概念所以可以并行进行多笔交易。假如你有多个UTXO你可以进行多笔交易而不需要并行锁。然后其还有匿名性的特征你可以隐藏自己的交易目的地通过设置的多个output而且没有余额意味着是没有状态的。要知道你有多少个比特币只需要把UTXO的交易记录统计一下就可以知道了。但这也让人比较费解而且也不利于应用上的开发。以太坊则使用了余额的方式。
在这篇文章中我先讲述了什么是区块链以及它的核心原理是什么。随后分享了比特币的hash算法以及Merkle Root是如何计算出来的。最后介绍了比特币的交易模型。希望对你有帮助。**这篇文章中图片很多,很难用音频体现出来,所以没有录制音频,还望谅解。**
文末给出了《区块链技术》系列文章的目录,希望你能在这个列表里找到自己感兴趣的内容。
- [区块链的革命性及技术概要](https://time.geekbang.org/column/article/5197)
- [区块链技术细节:哈希算法](https://time.geekbang.org/column/article/5363)
- [区块链技术细节:加密和挖矿](https://time.geekbang.org/column/article/5438)
- [去中心化的共识机制](https://time.geekbang.org/column/article/5612)
- [智能合约](https://time.geekbang.org/column/article/5623)
- [传统金融和虚拟货币](https://time.geekbang.org/column/article/5636)