| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 区块链 -> 区块链说数据没被篡改过,是不是在骗你 ? -> 正文阅读 |
|
[区块链]区块链说数据没被篡改过,是不是在骗你 ? |
【背景介绍】 这份数据证明是什么?又是怎么实现的?带着这份疑问,本文将详细介绍目前主流的数据证明的实现以及解决方案和优化思路。 【默克尔证明】 默克尔树(Merkle Tree),因发明人叫Merkle,并且是树形结构而得名。如下图,默克尔树的叶节点存储数据或者数据的哈希值,任一父节点包含了其子节点总和的哈希值。 默克尔树最大的作用便是快速校验部分数据是否在原始数据中。 举个例子,要想验证L2在上图的这棵树上,只需要节点列表[L2,Hash0-0,Hash1]即可,通过L2可以计算出Hash0-1,通过Hash0-1与Hash0-0可以计算出Hash0,再通过Hash0与Hash1,便可以得到一个树根,再把该树根与Top Hash进行比对即可快速验证L2是否在树中。默克尔树生成数据证明以及验证数据证明的原理就是如此,生成证明时自底向上,不断获取父节点的兄弟节点,打包成数据证明;验证时通过遍历证明中的节点列表,不断进行哈希得到父节点的操作,最后可以得到树根节点,再把该树根与原树根进行比对,达到验证的目的。 这种基本的默克尔树广泛应用在区块链系统中,例如比特币中轻钱包的SPV(简单支付验证)便是应用该原理,比特币中区块记录的root便是区块中交易集合构成的默克尔树的树根,使得无需下载完整区块,只需要区块头便可以验证某条交易的有效性。 【以太坊上的数据证明】 ▲ 什么是MPT? 基于默克尔树以及前缀树的优缺点,MPT提出了四种节点类型: 1)空节点:用来表示空字符串; 相比于普通的前缀树以及默克尔树,MPT加入了扩展节点,可以将不共享的前缀压进一个节点,带来路径压缩的特性,降低了树的高度,减少了空间浪费;分支节点的加入,使得单个节点最多能有十六个分叉,同样也是降低了树高;并且通过哈希值进行树节点间的连接,确保了安全性与可验证性。 对MPT上的数据进行增删改查,需要通过前缀索引到对应的节点,叶子节点的改动会带来新的分支节点或者扩展节点的加入,新的节点需要进行哈希计算得出新的哈希值,然后更新父节点的信息,造成索引路径上节点的哈希值的连锁的改动,最后生成一个唯一的“世界状态”。 在MPT上,要想生成数据证明,也需要沿着树进行前缀的索引,并把索引路径上的节点打包成证明返回即可。验证数据证明的时候,需要验证叶子节点中键的正确性,并且对证明中的节点列表进行自底向上的哈希操作,最后比对树根的值即可。 ▲ 性能瓶颈 由于以太坊底层采取的是leveldb存储,对于每个键值的读取最坏情况下需要一次磁盘IO,那么在执行合约的时候,将会有大量的时间耗费在状态的读取中,久而久之系统整体的性能会越来越差。 【联盟链下如何优化?】 1)每个节点会有一个ID; 如上图,将状态数据以键值对的形式存入数据节点,每个节点最后会以页的形式存放到磁盘中,为了更适应磁盘的读写策略,规定每个页大小为4K(磁盘页大小通常为4K),这样就可以通过计算出页ID*页大小的偏移量从磁盘读取某个节点。 同样我们规定每个节点最多容纳4K的数据,如果某个数据节点插入太多键值对,导致其超过了4K的阈值,将会产生分裂,变为多个不超过阈值的新数据节点,并把对应的新节点信息插入到原父节点的子节点列表中,再递归检查父节点即索引节点是否超过阈值,如果超过将会重复分裂过程。 这里我们会提供一个最大页ID以及空闲ID列表来管理页ID,当新建了一个节点时,会先去空闲ID列表查询一下是否存在满足条件的ID,有则取出来分配给新节点,否则会把最大页ID分配给新节点,并递增最大页ID。有分配自然就会有回收,当节点A分裂为节点B和节点C时,节点A的页ID就会被回收,纳入到空闲ID列表中。当发现某个节点变为空节点时,也会被回收页ID。 由于默克尔B+树在每个节点都存储了哈希值,每次对状态数据的增删改查,都会造成底层数据节点的变动,进而影响从数据节点到根节点路径上的节点哈希值,最后生成出唯一的“世界状态”。 在生成数据证明时,同样是沿着树根,通过键的比对向下索引,并把索引路径上的节点打包成证明返回即可。验证数据证明的时候,可以在数据证明中的节点路径进行同样的索引操作,验证索引路径的一致性,并且进行节点哈希值的正确性校验,最后比对树根的值即可。 ▲ 优化思路 根据上文来看,默克尔B+树确实可以实现较高的读写效率以及可验证性,但是还有很大的优化空间,基于此,我们打算从优化磁盘读的角度提出优化思路。 对于节点限制在4K大小这点,可以有效提高节点分叉数,降低树高,进而降低磁盘IO次数,但是仍然需要log(n)次的磁盘IO,为了最大化降低IO次数,我们把索引节点与数据节点分开存储为索引文件和数据文件。索引节点中可以认为只存放了键,因此索引文件远远小于数据文件,把索引文件通过mmap(内存映射)的形式维护在内存中,这样对底层的数据的读取只需要数据文件的一次磁盘IO即可,大大提升了读效率,在联盟链读多写少的场景下具有较大的优势。 【小结】 针对联盟链读多写少的场景,本文提出了一种以默克尔B+树为基础的解决方案,解释了在默克尔B+树上的数据证明以及验证的实现,并且为了进一步提高性能,提供了优化思路,通过索引与数据分离的方式来完善可用性。 作者简介 参考文献 |
|
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 | -2024/11/25 20:34:00- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |