C++之双向链表与哈希链表用法区别实例(二百六十八)

2024-04-06 05:44

本文主要是介绍C++之双向链表与哈希链表用法区别实例(二百六十八),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长!

优质专栏:Audio工程师进阶系列原创干货持续更新中……】🚀
优质专栏:多媒体系统工程师系列原创干货持续更新中……】🚀
优质视频课程:AAOS车载系统+AOSP14系统攻城狮入门实战课原创干货持续更新中……】🚀

人生格言: 人生从来没有捷径,只有行动才是治疗恐惧和懒惰的唯一良药.

更多原创,欢迎关注:Android系统攻城狮

欢迎关注Android系统攻城狮

1.前言

本篇目的:在阅读Linux内核代码时,发现Binder驱动中的双向链表和哈希链表的挺有意思,分享给大家。

2.双向链表与哈希链表介绍

  • 在Android的Binder区域中,双向链表(struct list_head)和哈希链表(struct hlist_node)是两种不同的数据结构,它们在实现中有一些区别和各自的作用。
  • 首先,让我们了解这两种数据结构的基本概念:
  1. 双向链表(struct list_head)

    • 双向链表是一种数据结构,其中每个节点都包含指向前一个节点和后一个节点的指针。这使得在双向链表中,节点可以双向遍历。
    • 在Linux内核中,双向链表是一种常见的数据结构,用于连接内核中的各种对象,例如进程、文件等。
    • 在Binder区域,双向链表常用于维护连接Binder驱动程序的客户端和服务端之间的通信通道。它们可以帮助管理Binder对象的生命周期和通信路线。
  2. 哈希链表(struct hlist_node)

    • 哈希链表是一种特殊的链表结构,它与传统链表不同之处在于,每个节点可能连接到一个链表桶中,而不是简单地按照顺序连接。
    • 在哈希链表中,节点被散列到特定的桶中,以提高检索效率。这使得在大型数据集中查找特定项的时间复杂度得到了显著改善。
    • 在Android的Binder区域,哈希链表通常用于管理Binder节点的分配和释放。通过哈希链表,系统可以更有效地管理Binder对象的分配和回收,以提高性能和资源利用率。

现在来比较双向链表和哈希链表在Binder区域中的作用和区别:

  1. 作用

    • 双向链表用于维护Binder通信通道的连接关系,帮助管理客户端和服务端之间的通信路线。
    • 哈希链表用于管理Binder对象的分配和释放,以提高资源管理的效率和性能。
  2. 区别

    • 双向链表提供了顺序访问节点的能力,而哈希链表则通过散列将节点分布到不同的桶中,提高了查找效率。
    • 双向链表的节点包含指向前一个和后一个节点的指针,而哈希链表的节点可能包含指向桶中下一个节点的指针。
  • 总的来说,在Android的Binder区域中,双向链表和哈希链表都是重要的数据结构,用于管理Binder对象和通信通道,但它们在实现方式和作用上略有不同,以满足不同的需求和优化性能。

Android的Binder驱动数据结构

  • 在Android的Binder驱动数据结构中,struct list_head 表示双向链表,而 struct hlist_head 和 struct hlist_node 表示哈希链表的头部和节点。

  • 1.双向链表 struct list_head

  • 每个 struct list_head 结构包含两个指针,分别指向下一个节点和前一个节点,因此构成了一个双向链表。

  • 用途:双向链表通常用于存储元素的有序集合,并且支持在常量时间内对链表中的元素进行插入、删除和前向/后向遍历。

  • 2.哈希链表 struct hlist_head 和 struct hlist_node

  • struct hlist_head 表示哈希链表的头部,其中包含一个指向链表的第一个节点的指针。

  • struct hlist_node 表示哈希链表中的节点,每个节点包含一个指向下一个节点的指针以及一个指向前一个节点指针的指针。

  • 用途:哈希链表通常用于在哈希表中解决哈希冲突。每个桶对应一个哈希链表,这样相同哈希值的键值对可以通过链表链接在一起。

3.代码实例

<1>.双向链表代码实例

#include <iostream>// 双向链表节点结构
struct list_head {struct list_head *next, *prev;
};int main() {// 创建双向链表struct list_head list;struct list_head node1, node2, node3;// 初始化双向链表头list.next = &node1;list.prev = &node3;// 初始化节点node1.next = &node2;node1.prev = &list;node2.next = &node3;node2.prev = &node1;node3.next = &list;node3.prev = &node2;// 遍历双向链表并打印节点地址struct list_head *pos;std::cout << "双向链表节点地址:";for (pos = list.next; pos != &list; pos = pos->next) {std::cout << pos << " ";}std::cout << std::endl;return 0;
}

<2>.哈希链表代码实例

#include <iostream>// 哈希链表头结构
struct hlist_head {struct hlist_node *first;
};// 哈希链表节点结构
struct hlist_node {struct hlist_node *next, **pprev;
};int main() {// 创建哈希链表struct hlist_head hash_table[10];struct hlist_node node1, node2, node3;// 初始化哈希链表头for (int i = 0; i < 10; ++i) {hash_table[i].first = NULL;}// 初始化节点node1.next = &node2;node1.pprev = &hash_table[0].first;node2.next = &node3;node2.pprev = &node1.next;node3.next = NULL;node3.pprev = &node2.next;// 遍历哈希链表并打印节点地址std::cout << "哈希链表节点地址:";struct hlist_node *hpos;for (hpos = hash_table[0].first; hpos != NULL; hpos = hpos->next) {std::cout << hpos << " ";}std::cout << std::endl;return 0;
}

这篇关于C++之双向链表与哈希链表用法区别实例(二百六十八)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C语言中%zu的用法解读

《C语言中%zu的用法解读》size_t是无符号整数类型,用于表示对象大小或内存操作结果,%zu是C99标准中专为size_t设计的printf占位符,避免因类型不匹配导致错误,使用%u或%d可能引发... 目录size_t 类型与 %zu 占位符%zu 的用途替代占位符的风险兼容性说明其他相关占位符验证示

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

全面解析Golang 中的 Gorilla CORS 中间件正确用法

《全面解析Golang中的GorillaCORS中间件正确用法》Golang中使用gorilla/mux路由器配合rs/cors中间件库可以优雅地解决这个问题,然而,很多人刚开始使用时会遇到配... 目录如何让 golang 中的 Gorilla CORS 中间件正确工作一、基础依赖二、错误用法(很多人一开

Java Stream流之GroupBy的用法及应用场景

《JavaStream流之GroupBy的用法及应用场景》本教程将详细介绍如何在Java中使用Stream流的groupby方法,包括基本用法和一些常见的实际应用场景,感兴趣的朋友一起看看吧... 目录Java Stream流之GroupBy的用法1. 前言2. 基础概念什么是 GroupBy?Stream

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

MySQL多实例管理如何在一台主机上运行多个mysql

《MySQL多实例管理如何在一台主机上运行多个mysql》文章详解了在Linux主机上通过二进制方式安装MySQL多实例的步骤,涵盖端口配置、数据目录准备、初始化与启动流程,以及排错方法,适用于构建读... 目录一、什么是mysql多实例二、二进制方式安装MySQL1.获取二进制代码包2.安装基础依赖3.清

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

Java Spring的依赖注入理解及@Autowired用法示例详解

《JavaSpring的依赖注入理解及@Autowired用法示例详解》文章介绍了Spring依赖注入(DI)的概念、三种实现方式(构造器、Setter、字段注入),区分了@Autowired(注入... 目录一、什么是依赖注入(DI)?1. 定义2. 举个例子二、依赖注入的几种方式1. 构造器注入(Con

SpringBoot 异常处理/自定义格式校验的问题实例详解

《SpringBoot异常处理/自定义格式校验的问题实例详解》文章探讨SpringBoot中自定义注解校验问题,区分参数级与类级约束触发的异常类型,建议通过@RestControllerAdvice... 目录1. 问题简要描述2. 异常触发1) 参数级别约束2) 类级别约束3. 异常处理1) 字段级别约束