Python 算法基础篇:哈希表与散列函数

news2025/1/18 20:13:10

Python 算法基础篇:哈希表与散列函数

  • 引用
  • 1. 哈希表的概念
  • 2. 散列函数的概念
    • a ) 一致性
    • b ) 均匀性
    • c ) 高效性
  • 3. 散列函数的实现
  • 4. 哈希表的实现
  • 5. 哈希表的冲突解决
    • a ) 链地址法
    • b ) 开放地址法
  • 6. 实例演示
    • 实例:电话簿
  • 总结

引用

哈希表是一种高效的数据结构,常用于存储键值对并支持快速的插入、查找和删除操作。散列函数是哈希表的关键组成部分,用于将键映射到哈希表的索引位置。本篇博客将介绍哈希表和散列函数的基本概念,并通过实例代码演示它们的应用。

😃😄 ❤️ ❤️ ❤️

1. 哈希表的概念

哈希表是一种数据结构,它将键值对存储在一个数组中,并通过散列函数将键映射到数组的索引位置。这样可以快速地插入、查找和删除键值对,使得哈希表成为一种高效的数据结构。哈希表的查找操作的平均时间复杂度为 O ( 1 ),在理想情况下可以达到常数时间。

哈希表的主要优点是快速的查找操作,但它也有一些局限性。首先,哈希表的键必须是可哈希的,即可以通过散列函数计算得到唯一的哈希值。其次,哈希表的内存消耗较大,因为需要维护一个数组来存储数据。最后,哈希表的查找操作在最坏情况下可能变得很慢,如果哈希函数导致冲突,多个键被映射到同一个索引位置,就需要处理冲突。

2. 散列函数的概念

散列函数是哈希表的关键组成部分,它将键映射到哈希表的索引位置。散列函数必须满足以下特性:

a ) 一致性

对于相同的键,散列函数应该始终返回相同的哈希值。这样可以确保相同的键在哈希表中总是存储在相同的位置,实现快速的查找操作。

b ) 均匀性

散列函数应该将键均匀地映射到哈希表的不同索引位置,减少冲突的发生。这样可以确保哈希表中的数据分布均匀,避免出现过多的冲突。

c ) 高效性

散列函数应该能够在常数时间内计算出哈希值,以保持快速的插入、查找和删除操作。

3. 散列函数的实现

Python 内置了一个 hash() 函数,它可以用于获取对象的哈希值。对于大多数内置类型, hash() 函数能够返回唯一的哈希值。例如,对于整数、浮点数和字符串等类型, hash() 函数都能返回唯一的哈希值。下面是一个示例代码:

# 使用hash()函数获取哈希值
print(hash(42))
print(hash(3.14))
print(hash('hello'))

代码解释:上述代码演示了 hash() 函数在整数、浮点数和字符串类型上的应用。对于整数和浮点数, hash() 函数能够返回唯一的哈希值;对于字符串,它也能返回唯一的哈希值。

然而,需要注意的是,用户自定义的对象默认情况下不支持 hash() 函数,因为 Python 不知道如何将用户自定义的对象映射到哈希表的索引位置。如果需要自定义散列函数,可以在对象的类中实现 __hash__() 方法。

4. 哈希表的实现

Python 中没有直接的哈希表数据结构,但我们可以使用字典( dictionary )来实现哈希表的功能。字典是 Python 中的一种内置数据结构,用于存储键值对。下面是一个示例代码:

# 创建字典
student_scores = {'Alice': 95, 'Bob': 80, 'Charlie': 75, 'David': 90}

# 查找元素
print("Alice 的成绩:", student_scores['Alice'])

# 插入元素
student_scores['Emma'] = 88

# 删除元素
del student_scores['Charlie']

# 打印字典
print("学生成绩表:", student_scores)

代码解释:上述代码演示了如何使用字典实现哈希表的功能。首先,我们创建了一个存储学生姓名和成绩的字典。通过使用键来查找元素,我们可以快速获取学生的成绩。然后,我们可以插入新的键值对和删除不需要的键值对。最后,打印字典的内容。

5. 哈希表的冲突解决

在散列函数的映射过程中,不同的键可能会产生相同的哈希值,这就是冲突。当出现冲突时,我们需要解决冲突,确保每个键能够正确地映射到哈希表的索引位置。

a ) 链地址法

链地址法是一种简单且常用的解决冲突的方法。它使用一个链表来存储哈希值相同的键值对。当发生冲突时,新的键值对会被添加到链表中,这样可以保证所有的键值对都能被正确地存储在哈希表中。

b ) 开放地址法

开放地址法是另一种解决冲突的方法。它在发生冲突时不使用链表,而是在哈希表中寻找下一个可用的空槽来存储键值对。有多种开放地址法的实现方式,如线性探测、二次探测和双重散列等。

6. 实例演示

现在,让我们通过一个实例来演示哈希表的应用,以及使用链地址法解决冲突。

实例:电话簿

假设我们需要实现一个电话簿应用,存储人名和对应的电话号码。下面是一个示例代码:

class HashTable:
    def __init__(self, size):
        self.size = size
        self.table = [[] for _ in range(size)]

    def _hash_function(self, key):
        return hash(key) % self.size

    def insert(self, key, value):
        index = self._hash_function(key)
        self.table[index].append((key, value))

    def search(self, key):
        index = self._hash_function(key)
        for k, v in self.table[index]:
            if k == key:
                return v
        return None

    def delete(self, key):
        index = self._hash_function(key)
        for i, (k, v) in enumerate(self.table[index]):
            if k == key:
                del self.table[index][i]
                return

# 创建电话簿
phone_book = HashTable(10)

# 添加联系人信息
phone_book.insert('Alice', '123456789')
phone_book.insert('Bob', '987654321')
phone_book.insert('Charlie', '456789123')

# 查找联系人电话号码
print("Alice 的电话号码:", phone_book.search('Alice'))

# 删除联系人信息
phone_book.delete('Bob')

# 打印电话簿
print("电话簿内容:", phone_book.table)

代码解释:上述代码演示了使用哈希表实现电话簿应用的示例。我们创建了一个 HashTable 类来表示哈希表,其中包括插入、查找和删除操作的实现。我们通过散列函数将人名映射到哈希表的索引位置,并使用链地址法解决冲突,确保人名和电话号码正确地存储在哈希表中。

总结

本篇博客介绍了哈希表和散列函数的基本概念,并通过实例代码演示了它们的应用。哈希表是一种高效的数据结构,用于存储键值对并支持快速的插入、查找和删除操作。散列函数是哈希表的关键组成部分,用于将键映射到哈希表的索引位置。

[ 专栏推荐 ]
😃 Python 算法初阶:入门篇》😄
❤️【简介】:本课程是针对 Python 初学者设计的算法基础入门课程,涵盖算法概念、时间复杂度、空间复杂度等基础知识。通过实例演示线性搜索、二分搜索等算法,并介绍哈希表、深度优先搜索、广度优先搜索等搜索算法。此课程将为学员提供扎实的 Python 编程基础与算法入门,为解决实际问题打下坚实基础。

在这里插入图片描述

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

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

相关文章

[计算机入门] 文件夹(目录)及路径

2.7 文件夹(目录)及路径 文件夹(folder、目录)是一种用于存储和组织文件和其他文件夹的容器。它可以包含任意数量的文件和子文件夹,并且可以通过拖放、复制和粘贴等操作来移动、复制和管理这些文件和子文件夹。 Windows文件夹可以帮助用户更…

spring复习:(43)使用TransactionProxyFactoryBean来实现事务时,事务是怎么开启的?

一、配置文件&#xff1a; <bean id"myFactoryBean"class"org.springframework.transaction.interceptor.TransactionProxyFactoryBean"><property name"transactionManager" ref"transactionManager" /><property nam…

可以替代微软 Exchange 的 几个开源软件分享给你

微软Exchange是一个功能强大的邮件和群件解决方案&#xff0c;但对于一些用户来说&#xff0c;寻找替代方案可能是必要的。幸运的是&#xff0c;有几个开源软件提供了可靠而且功能丰富的替代选项。这些开源软件不仅可以满足组织和个人的邮件和协作需求&#xff0c;还具有灵活性…

quartus18.0如何下载安装Cyclone V器件库

文章目录 前言一、下载流程二、添加步骤三、总结四、参考资料 前言 在我们使用不同版本的板子的时候&#xff0c;我们需要在quartus下安装不同型号的器件库才能对板子进行选型并进行下一步操作。 一、下载流程 官网下载地址 这里我们点击支持选中下载中心&#xff1a; 选择FPGA…

reggie优化04-Nginx

官方网站下载&#xff1a;http://nginx.org/en/download.html 1、Nginx安装 这里需要在Linux系统下&#xff1a; 安装wget工具&#xff1a;yum install wget&#xff08;或者官网下载直接上传到Linux&#xff09; 安装树形结构tree&#xff1a;yum install tree 2、Nginx命令 …

在云计算环境中,保护Java应用程序可用的有效措施和工具

云计算&#xff08;Cloud&#xff09;技术是近年来计算机科学的一个重要突破。大多数组织已经通过将自己的应用程序移入云平台而获益。不过&#xff0c;如何保证应用程序在第三方服务器上的安全性&#xff0c;是一项艰巨的挑战。 在本文中&#xff0c;我们将重点讨论Java&…

Notepad++ 配置python虚拟环境(Anaconda)

Notepad配置python运行环境步骤&#xff1a; 打开Notepad ->”运行”菜单->”运行”按钮在弹出的窗口内输入以下命令&#xff1a; 我的conda中存在虚拟环境 (1) base (2) pytorch_gpu 添加base环境至Notepad中 cmd /k chdir /d $(CURRENT_DIRECTORY) & call cond…

《零基础入门学习Python》第036讲:类和对象:给大家介绍对象

0. 请写下这一节课你学习到的内容&#xff1a;格式不限&#xff0c;回忆并复述是加强记忆的好方式&#xff01; &#xff08;一&#xff09;对象 这节课给大家介绍对象。我们之前说过Python无处不对象&#xff0c;Python到处都是对象&#xff0c;然而我们很多人不理解对象到底…

NB!更方便Xshell本地密码破解工具

工具介绍 XshellCrack是基于SharpXDecrypt的二次开发&#xff0c;用go语言重写&#xff0c;增加了注册表查询设置&#xff0c;更方便xshell本地密码破解。 关注【Hack分享吧】公众号&#xff0c;回复关键字【230717】获取下载链接 工具使用 Usage:root SshCrack [flags]Flags…

在线看板工具Restyaboard

本文软件由网友 yf33 推荐&#xff1b; 什么 Restyaboard &#xff1f; Restyaboard 是一款类 Trello 应用&#xff0c;支持看板、任务、待办事项、聊天等。Restyaboard 的面板能为您提供项目当前状态的视觉概览&#xff0c;并通过让您专注于最重要的几个项目来提高您的工作效率…

FiddlerScript修改指定参数的返回值

FiddlerScript修改指定参数的返回值 使用场景&#xff1a; api/Live/GetLiveList接口&#xff1a; &#xff08;1&#xff09;Type为1&#xff0c;接口返回直播列表 &#xff08;2&#xff09;Type为2&#xff0c;接口返回回放列表 现在想修改直播列表的返回值 思路&#…

【Redis】6、Redisson 分布式锁的简单使用(可重入、重试机制...)

目录 零、自己通过 set nx ex 实现的分布式锁存在的问题一、Redisson 介绍二、Redisson 基本使用&#xff08;改造业务&#xff09;(1) 依赖(2) 配置 Redisson 客户端(3) 使用 Redisson 的可重入锁 三、Redisson 可重入锁原理四、Redisson 可重试原理五、Redisson 超时释放&…

线数据的按节点打断

思想&#xff1a;运行要素转线工具箱 原始数据 运行完数据 数量由7变成27

Springboot + Vue 下载Word、PDF文档并保留内部格式

相对于上传&#xff0c;下载时复杂的地方就是文件中文名称乱码 前端 <el-button click"clickCall(handleExport, scope,index)">导出</el-button>// 文件下载操作handleExport(row) {axios.get(**********master/proj/exportContract?id" row.id,…

前端node.js入门

(创作不易&#xff0c;感谢有你&#xff0c;你的支持&#xff0c;就是我前行的最大动力&#xff0c;如果看完对你有帮助&#xff0c;请留下您的足迹&#xff09; 目录 Node.js 入门 什么是 Node.js&#xff1f; 什么是前端工程化&#xff1f; Node.js 为何能执行 JS&…

netty组件详解-上

netty服务端示例: private void doStart() throws InterruptedException {System.out.println("netty服务已启动");// 线程组EventLoopGroup group new NioEventLoopGroup();try {// 创建服务器端引导类ServerBootstrap server new ServerBootstrap();// 初始化服…

CDHD高创驱动器通过ServoStudio备份和恢复参数的具体方法步骤

CDHD高创驱动器通过ServoStudio备份和恢复参数的具体方法步骤 硬件连接: 如下图所示,通过通信线缆将伺服驱动器和电脑进行连接,一端为RJ11,一端为USB, 软件连接: 打开伺服调试软件ServoStudio,在驱动器配置中找到连接—自动连接,点击搜索&连接,此时软件会自动搜索…

基于jsp+sevlet+mysql实验室设备管理系统2.0

基于jspsevletmysql实验室设备管理系统2.0 一、系统介绍二、功能展示1.控制台2.申购设备3.设备列表4.设备维护5.设备类型6.报废设备7.维修记录 四、其他系统实现五、获取源码 一、系统介绍 系统主要功能&#xff1a; 普通用户&#xff1a;控制台、申购设备、设备列表、设备维护…

docker 安装oracle 11g

docker 安装oracle 11g 拉取镜像 docker pull registry.cn-hangzhou.aliyuncs.com/helowin/oracle_11g创建容器 docker run -d -p 1521:1521 --name oracle11g registry.cn-hangzhou.aliyuncs.com/helowin/oracle_11gD:\docker\oracle\oracle11g>docker exec -it oracle11…

CMU 15-445 -- Query Optimization - 10

CMU 15-445 -- Query Optimization - 10 引言Query Optimization TechniquesQuery RewritingPredicate PushdownProjections Pushdown Cost-based SearchCost EstimationStatisticsEquality PredicateRange PredicateNegation QueryConjunction QueryDisjunction QueryJoins直方…