p2p、分布式,区块链笔记: Merkle-DAG和Merkle-Tree的区别与联系

2024-09-01 14:04

本文主要是介绍p2p、分布式,区块链笔记: Merkle-DAG和Merkle-Tree的区别与联系,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Merkle-DAG和Merkle-Tree的区别与联系

  1. 结构:

    • Merkle-Tree 是一种二叉树结构,每个非叶子节点是其子节点哈希的哈希。它具有层次结构,通常用于验证数据的完整性。
    • Merkle-DAG(有向无环图)是一种更通用的图结构,其一个节点可以有多个父节点和子节点。它允许更复杂的链接关系和非线性结构,适用于记录和追踪变更,支持广泛的并行操作和高效的增量更新。Merkle DAG 类似于 Merkle 树,但没有余额要求,每个节点都可以携带一个有效载荷。在 DAG 中,多个分支可以重新收敛,或者换句话说,一个节点可以有多个父节点。
  2. 用途:

    • Merkle-Tree 通常用于区块链和文件系统中以验证数据块的完整性(如 Bitcoin 和 Git)。
    • Merkle-DAG 用于数据去重和版本控制(如 IPFS),支持更灵活的数据组织和高效的同步。

CODE

  • 项目地址:https://github.com/zaphar/merkle-dag/blob/main/src/dag/mod.rs

NODE结构体

  • node.rs中定义了一个在Merkle DAG(有向无环图)中使用的节点结构体(Node)。每个节点包含一个有效载荷(item)和一组依赖ID(dependency_ids)。节点的唯一标识符(id)是由有效载荷和依赖ID的字节组合而成,确保相同的有效载荷和依赖ID总是有相同的ID。
// 代码位置 https://github1s.com/zaphar/merkle-dag/blob/main/src/node.rs#L51-L62
#[derive(Debug, PartialEq, Eq, Serialize, Deserialize)]
#[serde(from = "NodeSerde")]
pub struct Node<HW> // <HW> 为泛型参数
whereHW: HashWriter, //  where为泛型约束,泛型参数 HW 必须实现 HashWriter trait
{id: Vec<u8>,item: Vec<u8>,item_id: Vec<u8>,dependency_ids: BTreeSet<Vec<u8>>,_phantom: PhantomData<HW>,
}

add_node函数

函数签名

pub fn add_node<'a, N: Into<Vec<u8>>>(&'a mut self,item: N,dependency_ids: BTreeSet<Vec<u8>>,
) -> Result<Vec<u8>>
  • 'a: 生命周期标注,表明函数借用的是 self 的生命周期。
  • N: Into<Vec<u8>>: 泛型参数 N,它可以转换为 Vec<u8>
  • item: 节点的负载数据,将被转换为 Vec<u8>
  • dependency_ids: 节点的依赖 ID 集合,每个 ID 是 Vec<u8> 类型。
  • Result<Vec<u8>>: 函数返回一个结果,如果成功,返回新节点的 ID;如果失败,返回错误。

函数解释

  • add_node 函数负责将一个带有依赖关系的新节点添加到 Merkle-DAG 中。它首先检查节点是否已经存在,处理依赖关系,并更新节点存储和根集合。该函数确保节点的添加操作是幂等的,即对于相同的输入,结果不会发生变化。

在这里插入图片描述

/// 添加一个新负载及其依赖 ID 集合。此方法将构建一个新节点,并将其添加到 DAG 中。
/// 对于任何给定的输入,该方法是幂等的。
///
/// 不创建节点然后再添加的一个结果是,确保始终满足 merkle-crdt 白皮书中的实现规则。
pub fn add_node<'a, N: Into<Vec<u8>>>(&'a mut self,item: N,dependency_ids: BTreeSet<Vec<u8>>,
) -> Result<Vec<u8>> {// 使用提供的负载和依赖 ID 创建一个新节点实例。let node = Node::<HW>::new(item.into(), dependency_ids.clone());// 获取新创建的节点的 ID。let id = node.id().to_vec();// 检查节点是否已经存在于存储中。if self.nodes.contains(id.as_slice())? {// 如果节点已存在,则返回该节点的 ID。return Ok(self.nodes.get(id.as_slice()).unwrap().unwrap().id().to_vec());}// 初始化一个向量,用于记录需要从根集合中移除的节点。let mut root_removals = Vec::new();// 遍历所有的依赖 ID。for dep_id in dependency_ids.iter() {// 检查每个依赖节点是否存在于存储中。if !self.nodes.contains(dep_id)? {// 如果任何依赖节点不存在,则返回错误。return Err(StoreError::NoSuchDependents);}// 如果某个依赖 ID 存在于根集合中,则将其标记为需要移除。if self.roots.contains(dep_id) {root_removals.push(dep_id);}}// 更新根集合// 将新节点存储到节点存储中。self.nodes.store(node)?; // https://github1s.com/zaphar/merkle-dag/blob/main/src/store.rs// 从根集合中移除被标记的节点。for removal in root_removals {self.roots.remove(removal);}// 将新节点的 ID 添加到根集合中。self.roots.insert(id.to_vec());// 返回新节点的 ID。Ok(id.to_vec())
}

函数调用

  1. TestDag别名类型:

    // https://github1s.com/zaphar/merkle-dag/blob/main/src/test.rs#L19-L22
    type TestDag<'a> = Merkle<BTreeMap<Vec<u8>, Node<std::collections::hash_map::DefaultHasher>>,std::collections::hash_map::DefaultHasher,
    >;
    
    • Merkle 的第一个泛型参数是 BTreeMap<Vec<u8>, Node<std::collections::hash_map::DefaultHasher>>,它是一个键为 Vec<u8>,值为 Node<DefaultHasher> 的映射。
    • 第二个泛型参数是 DefaultHasher,它是一个标准库提供的哈希算法。
  2. 测试函数:

    // https://github1s.com/zaphar/merkle-dag/blob/main/src/test.rs#L26-L27
    #[test]
    fn test_root_pointer_hygiene() {let mut dag = TestDag::new(BTreeMap::new());let quax_node_id = dag.add_node("quax", BTreeSet::new()).unwrap();
    }
    
    • #[test] 属性标记此函数为测试函数。
    • dag 通过 TestDag::new(BTreeMap::new()) 创建一个新的 TestDag 实例,传入一个空的 BTreeMap
    • quax_node_id 调用 dag.add_node("quax", BTreeSet::new()) 方法添加一个新节点,节点 ID 被赋值给 quax_node_idunwrap() 处理可能出现的错误,确保成功添加节点。

CG

  • IPFS白皮书
  • DAG Builder 可视化工具
  • Merkle DAG implementation for IPLD in Python
  • 基于 Merkle-DAG 的并发版本历史系统实现

这篇关于p2p、分布式,区块链笔记: Merkle-DAG和Merkle-Tree的区别与联系的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1127139

相关文章

Vue和React受控组件的区别小结

《Vue和React受控组件的区别小结》本文主要介绍了Vue和React受控组件的区别小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录背景React 的实现vue3 的实现写法一:直接修改事件参数写法二:通过ref引用 DOMVu

Go之errors.New和fmt.Errorf 的区别小结

《Go之errors.New和fmt.Errorf的区别小结》本文主要介绍了Go之errors.New和fmt.Errorf的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考... 目录error的基本用法1. 获取错误信息2. 在条件判断中使用基本区别1.函数签名2.使用场景详细对

Redis实现分布式锁全过程

《Redis实现分布式锁全过程》文章介绍Redis实现分布式锁的方法,包括使用SETNX和EXPIRE命令确保互斥性与防死锁,Redisson客户端提供的便捷接口,以及Redlock算法通过多节点共识... 目录Redis实现分布式锁1. 分布式锁的基本原理2. 使用 Redis 实现分布式锁2.1 获取锁

Redis中哨兵机制和集群的区别及说明

《Redis中哨兵机制和集群的区别及说明》Redis哨兵通过主从复制实现高可用,适用于中小规模数据;集群采用分布式分片,支持动态扩展,适合大规模数据,哨兵管理简单但扩展性弱,集群性能更强但架构复杂,根... 目录一、架构设计与节点角色1. 哨兵机制(Sentinel)2. 集群(Cluster)二、数据分片

一文带你迅速搞懂路由器/交换机/光猫三者概念区别

《一文带你迅速搞懂路由器/交换机/光猫三者概念区别》讨论网络设备时,常提及路由器、交换机及光猫等词汇,日常生活、工作中,这些设备至关重要,居家上网、企业内部沟通乃至互联网冲浪皆无法脱离其影响力,本文将... 当谈论网络设备时,我们常常会听到路由器、交换机和光猫这几个名词。它们是构建现代网络基础设施的关键组成

Python学习笔记之getattr和hasattr用法示例详解

《Python学习笔记之getattr和hasattr用法示例详解》在Python中,hasattr()、getattr()和setattr()是一组内置函数,用于对对象的属性进行操作和查询,这篇文章... 目录1.getattr用法详解1.1 基本作用1.2 示例1.3 原理2.hasattr用法详解2.

Redis分布式锁中Redission底层实现方式

《Redis分布式锁中Redission底层实现方式》Redission基于Redis原子操作和Lua脚本实现分布式锁,通过SETNX命令、看门狗续期、可重入机制及异常处理,确保锁的可靠性和一致性,是... 目录Redis分布式锁中Redission底层实现一、Redission分布式锁的基本使用二、Red

redis和redission分布式锁原理及区别说明

《redis和redission分布式锁原理及区别说明》文章对比了synchronized、乐观锁、Redis分布式锁及Redission锁的原理与区别,指出在集群环境下synchronized失效,... 目录Redis和redission分布式锁原理及区别1、有的同伴想到了synchronized关键字

分布式锁在Spring Boot应用中的实现过程

《分布式锁在SpringBoot应用中的实现过程》文章介绍在SpringBoot中通过自定义Lock注解、LockAspect切面和RedisLockUtils工具类实现分布式锁,确保多实例并发操作... 目录Lock注解LockASPect切面RedisLockUtils工具类总结在现代微服务架构中,分布

JAVA覆盖和重写的区别及说明

《JAVA覆盖和重写的区别及说明》非静态方法的覆盖即重写,具有多态性;静态方法无法被覆盖,但可被重写(仅通过类名调用),二者区别在于绑定时机与引用类型关联性... 目录Java覆盖和重写的区别经常听到两种话认真读完上面两份代码JAVA覆盖和重写的区别经常听到两种话1.覆盖=重写。2.静态方法可andro