Python数据结构:堆的实现(python数据结构和算法分析)
off999 2024-11-09 12:53 15 浏览 0 评论
在本文中,我们将了解 Python 中的堆是什么以及怎样实现它。我们将通过最小堆的 python 程序实现来理解堆的概念。最后,我们将学习堆数据结构的时间复杂度和应用。那么,让我们开始吧!
什么是堆?
堆是一种遵循“完全”二叉树属性并满足堆属性的数据结构。因此,它也被称为二叉堆。完全二叉树是每一层都被填满,并且所有节点都尽可能靠左的树。在二叉树中,有可能最后一层是空的并且没有被填充。在堆数据结构中,我们为树的每个节点分配键值或权重。将根节点键值与子节点进行比较,然后根据比较大小将树相应地排列为两类,即最大堆和最小堆。堆数据结构可以用作堆排序算法来对数组或列表中的元素进行排序。堆排序算法可用于优先队列、订单统计、Prim 算法或Dijkstra 算法等。简而言之,堆数据结构在要重复删除最高或最低优先级对象时经常被使用。
建堆-Heapify?
首先我们需要了解什么是 heapify。使用二叉树创建堆数据结构的过程称为 Heapify。heapify 过程用于创建 Max-Heap 或 Min-Heap。让我们使用下面的示例来研究 Heapify:
考虑如下图所示的输入数组:
使用这个数组,我们将创建完整的二叉树。 我们从最后一个非叶子节点 (len(array)//2-1) 开始,将其作为当前的节点。如果要创建Min-Heap,我们要保证任何当前节点小于他的两个子节点。设当前节点的序号是k,那么其左子节点的序号是2k+1,右子节点是2k+2。Heapify就是要保证上述的局部性质,首先完成父节点的heapify,还要沿着一条树的路径递归完成子节点的heapify。接下来就是倒着数组序号进行Heapify,这样就完成了整个数组的堆化。数组在堆化过程中是以如下方式变化的:[3, 9, 2, 1, 4, 5]--> [3, 1, 2, 9, 4, 5]--> [3, 1, 2, 9, 4, 5]--> [1, 3, 2, 9, 4, 5]。以下程序演示了怎样heapify一个数组。时间复杂度是O(nlogn)
def min_heapify(A,k):
print(A)
l = left(k)
r = right(k)
if l < len(A) and A[l] < A[k]:
smallest = l
else:
smallest = k
if r < len(A) and A[r] < A[smallest]:
smallest = r
if smallest != k:
A[k], A[smallest] = A[smallest], A[k]
min_heapify(A, smallest)
def left(k):
return 2 * k + 1
def right(k):
return 2 * k + 2
def build_min_heap(A):
n = int((len(A)//2)-1)
for k in range(n, -1, -1):
min_heapify(A,k)
A = [3,9,2,1,4,5]
build_min_heap(A)理解min-heapify函数
此函数可以将节点及其所有后代(子节点及其子节点)遵循堆属性。它通过交换节点的键值来重新组织堆里的数据,使得当前节点成为其子树中的最小节点,遵循堆属性。
该函数首先在给定节点及其子节点中找到具有最小值的节点。然后它将给定节点(比如 i)与找到的最小值节点(比如 j)交换,然后在节点 j 上(递归地)调用 min-heapify 函数,以确保分配给节点 j 的新值确实不要破坏其子树中的堆属性。由于最多要遍历树的深度,所以它的时间复杂度是O(d),其中d是深度,或者,就节点数而言,O(log n),n是堆中的元素。
退出堆顶元素:heappop函数
该函数弹出堆的最小值(根元素)。
这实际上是通过将根节点与最后一个节点交换并删除现在的最后一个节点(包含最小值)然后为根节点调用 min-heapify 以在由于交换引起的更改后维护堆属性来完成的。
由于我们只需要调用一次min-heapify,因此时间复杂度为 O(log n),其中 n 是元素的数量,或者 O(h),其中 h 是树的高度,即 log n。
加入新元素:heappush 函数
此函数将一个新元素推入堆中,并将其排列到正确的位置,同时保持堆属性。
这实际上是通过在堆的末尾添加一个新节点来完成的。现在为了维护堆属性,我们从最后一个节点向上遍历(并在需要的地方交换)以修复可能被违反的堆属性。
与 heappop 类似,这里的时间复杂度是 O(log n),因为我们只需要遍历子树的高度。
获得最小值:extractMin 函数
此函数从堆中返回最高优先级(根元素)。由于我们只需要返回根的值而不对堆进行任何更改,并且根在 O(1) 时间内可以访问,因此函数的时间复杂度为 O(1)。
import sys
#defining a class min_heap for the heap data structure
class min_heap:
def __init__(self, sizelimit):
self.sizelimit = sizelimit
self.cur_size = 0
self.Heap = [0]*(self.sizelimit + 1)
self.Heap[0] = sys.maxsize * -1
self.root = 1
# helper function to swap the two given nodes of the heap
# this function will be needed for heapify and insertion to swap nodes not in order
def swapnodes(self, node1, node2):
self.Heap[node1], self.Heap[node2] = self.Heap[node2], self.Heap[node1]
# THE MIN_HEAPIFY FUNCTION
def min_heapify(self, i):
# If the node is a not a leaf node and is greater than any of its child
if not (i >= (self.cur_size//2) and i <= self.cur_size):
if (self.Heap[i] > self.Heap[2 * i] or self.Heap[i] > self.Heap[(2 * i) + 1]):
if self.Heap[2 * i] < self.Heap[(2 * i) + 1]:
# Swap the node with the left child and then call the min_heapify function on it
self.swapnodes(i, 2 * i)
self.min_heapify(2 * i)
else:
# Swap the node with right child and then call the min_heapify function on it
self.swapnodes(i, (2 * i) + 1)
self.min_heapify((2 * i) + 1)
# THE HEAPPUSH FUNCTION
def heappush(self, element):
if self.cur_size >= self.sizelimit :
return
self.cur_size+= 1
self.Heap[self.cur_size] = element
current = self.cur_size
while self.Heap[current] < self.Heap[current//2]:
self.swapnodes(current, current//2)
current = current//2
# THE HEAPPOP FUNCTION
def heappop(self):
last = self.Heap[self.root]
self.Heap[self.root] = self.Heap[self.cur_size]
self.cur_size -= 1
self.min_heapify(self.root)
return last
# THE BUILD_HEAP FUNCTION
def build_heap(self):
for i in range(self.cur_size//2, 0, -1):
self.min_heapify(i)
# helper function to print the heap
def print_heap(self):
for i in range(1, (self.cur_size//2)+1):
print("Parent Node is "+ str(self.Heap[i])+" Left Child is "+ str(self.Heap[2 * i]) + " Right Child is "+ str(self.Heap[2 * i + 1]))
# Driver Code
minHeap = min_heap(10)
minHeap.heappush(15)
minHeap.heappush(7)
minHeap.heappush(9)
minHeap.heappush(4)
minHeap.heappush(13)
minHeap.print_heap()相关推荐
- 不用拉网线的路由器是真的吗
-
是真的不插卡不拉线有线就有网,这11个字其实就涵盖了无线路由器的特点,无线路由器免插卡、不用拉网线,完全摆脱了之前家用路由器和网线捆绑的模式,有电就有网,其实说的就是无线路由器的使用操作简单,通电就可...
- u盘检测软件下载(u盘测试软件)
-
1、u盘芯片检测工具(ChipEasy)可以查看USB设备PID、VID、SN、制造商、产品名等;2、查看USB设备主控芯片信息、闪存芯片信息、固件信息、电流控制3、SSD型号...
- 电脑现在什么系统最好(电脑现在用什么系统好)
-
WINXP好用,但过时了。VISTA不好用,没推开就夭折了。WIN8/8.1是针对触模屏设计的,如果你用的不是触摸屏平板电脑是普通电脑,使WIN8/8.1总觉着很蹩扭。新出的WIN10,功能...
- 账号怎么注册(steam账号怎么注册)
-
如果注册是qq账号【qq号码的申请办法】【1】双击qq登陆界面,在qq帐号填写空格的后面你可以看见:[申请帐号];【2】点击[申请帐号]进入,就可以在网上免费申请号码了;【3】进入www.qq.com...
- tmp文件是什么意思(tmp文件有什么用)
-
在系统C:\Windows\Temp文件夹中,我们经常会发现一些后缀名为TMP的文件,在该文件夹中的这些文件其实都是临时文件。它们可能是系统被误关机,或者其他程序没有删除而生的。而且在该文件夹中还有其...
- 怎么给u盘格式化(怎么给u盘格式化成FAT32)
-
u盘插入电脑,等待桌面弹出u盘图标。打开“计算机”。左键选中u盘,单击右键,在弹出的菜单中,点击“格式化”。点击“开始”,点击“确定”即可。格式化u盘详细步骤1、找到U盘盘符,鼠标右键点击,弹出菜单中...
- harmonyos主题下载(harmonyos主题怎么换)
-
首先,打开荣耀手机的应用市场,在搜索框中输入“华为鸿蒙主题”,然后点击搜索。找到“华为鸿蒙主题”应用后,点击下载即可。下载完成后,打开“华为鸿蒙主题”应用,选择心仪的主题,点击下载并应用即可享受华为鸿...
- 戴尔笔记本电脑黑屏却开着机
-
对于电脑黑屏的处理基本上采用排除、替换相结合的方法,其原则应本着先替换排除可疑性最大的部件。对于普通电脑用户来讲,专业知识不足,可以按下列步骤分析故障原因,以便可以自己动手排除故障。首先检查接触是否良...
- 手机版电脑桌面下载(手机电脑桌面下载软件安装包)
-
只有电脑版手机助手软件,没有手机桌面这个软件在电脑上点击今日头条APP下载安装即可哦你好,陌陌电脑版如果说你想要下载到电脑桌面的话,你只需要长按把它添加到你的电脑桌面就可以了。要将软件下载到桌面并创建...
- ghost备份中文图解(ghost备份1837)
-
其实是这样的ghost文件备份后会生成两个文件一个是.GHO一个是.GHS文件FAT32格式的分区,单个文件最大只支持到2G(2048M),如果你的镜像>2G,这时的做的GHOST在一个文件里装...
- win10一键重装win7(win10一键重装系统)
-
1、首先准备一个4GB以上可以正常使用的U盘。2、在一个可以正常使用的电脑上,下载老毛桃软件并安装。3、去网上下载所需的win7,win10选择自己所需要的系统,并下载下来。4、插入u盘并打开老毛桃...
- 戴尔按f12还原系统步骤win10
-
基本上正常的话是f8,如果你希望他变成年,F12,你要打开设置去连,然后把这个快捷键的位置调一调戴尔的键盘f1到f12恢复原功能的方法:1、可能是操作者操作有误的原因,使键盘没有任何反应。2、根据复合...
-
- qq网页版官网(qq1网页版)
-
https://aq.qq.com/cn2/indexQQ安全中心是腾讯公司推出的QQ帐号保护软件,为广大QQ用户提供一站式的QQ安全服务,包括了密保管理、帐号保护、安全体检、修改密码、帐号申诉等功能,让账号更加安全可靠。为了全面保护QQ帐...
-
2025-11-18 15:03 off999
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
python入门到脱坑 输入与输出—str()函数
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
宝塔面板如何添加免费waf防火墙?(宝塔面板开启https)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
慕ke 前端工程师2024「完整」
-
失业程序员复习python笔记——条件与循环
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
