用调整法和插入法建堆的Python实现,不同建堆方式对堆排序性能的影响

本文主要是介绍用调整法和插入法建堆的Python实现,不同建堆方式对堆排序性能的影响,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

       建堆的方式大体有两种,一种是插入法,一种是调整法,其中以调整法比较常见。由于他们的建堆的思路不同,所以两种方法建堆结果可能不一样。

       插入法建堆是将数组1中的元素逐个插入到数组2中建立一个堆。以小根堆为例,每插入一个关键字就与其父节点的关键字比较大小,如果父节点的关键字较大则交换,然后依次自底地向上调整使之符合小根堆的特性。在某棵已插入根节点的子树中,当插入左节点时,左节点与根节点交换,插入右节点时,右节点与根节点交换,那么这种情况下这颗子树三个节点的位置都发生改变了。因此插入法建堆结果与插入的顺序和值大小有关。

       调整法建堆是自底向上依次调整,一棵子树中最小的节点值与根节点交换,最大的那个节点位置在本次调整中不作改变。即这种方法建堆有一个节点位置不变。

        下面是插入法和调整法建堆的Python代码:

import random
import datetime#make random file
def creatIntRandom():count = 100output = open('data_100.txt','w+')while count:output.write(str(random.randint(0,1000001)) + '\n')count = count - 1output.closedef txtToList():int_list = []in_file = open('data.txt')in_text = in_file.readlines()   for line in in_text:num = int(line[0 : len(line) - 1])int_list.append(num) in_file.close()return int_list#----------------------------------
def minHeapify(list, heapsize, index):left = 2*index + 1right = 2*index + 2mini = indexif left < heapsize:if list[mini] > list[left]:mini = leftif right < heapsize and list[mini] > list[right]:mini = rightif mini != index:list[mini], list[index] = list[index], list[mini]minHeapify(list, heapsize, mini)def buildMinHeap_1(list):heapSize = len(list)if heapSize < 2:returnfor i in range(heapSize/2 - 1, -1, -1):minHeapify(list, heapSize, i)def heapSort_1(list):buildMinHeap_1(list)for i in range(len(list) - 1, -1, -1):list[0], list[i] = list[i], list[0]minHeapify(list, i, 0)return list    #-------------------------------------------
def buildMinHeap_2(list_1):heapsize = 0list_2 = [0]*(len(list_1) + 1)for i in range(len(list_1)):heapsize = i + 1list_2[heapsize] = list_1[i]while heapsize > 2 and list_2[heapsize/2] > list_2[heapsize]:list_2[heapsize], list_2[heapsize/2] = list_2[heapsize/2], list_2[heapsize]heapsize/=2return list_2[1:len(list_2)]def heapSort_2(list):list_2 = buildMinHeap_2(list)for i in range(len(list)-1, -1, -1):list_2[0], list_2[i] = list_2[i], list_2[0]minHeapify(list_2, i, 0)return list_2#------------------------------------------
def verify(list):for i in range(len(list) - 1):if list[i] >= list[i+1]:passelse:return Falsereturn Truedef test():list_in = txtToList()time_start_1 = datetime.datetime.now()list_out_1 = heapSort_1(list_in)time_end_1 = datetime.datetime.now()#print list_out_1print verify(list_out_1)print (time_end_1 - time_start_1)time_start_2 = datetime.datetime.now()list_out_2 = heapSort_2(list_in)time_end_2 = datetime.datetime.now()#print list_out_2print verify(list_out_1)print (time_end_2 - time_start_2)#creatIntRandom()
test()


        下图是对100W个随机产生的整数进行堆排序的运行结果:

    

 


        调整法的时间复杂度:建堆耗时0.5NlogN,排序耗时NlogN,累计1.5NlogN。
        插入法的时间复杂度:建堆耗时NlogN,排序耗时NlogN,累计2NlogN。
        从以上的比较中可以看出,虽然两种方法的时间复杂度在数量级是一样的,但是由于建堆所耗费的时间不同,总体时间有所不同,从上图执行的结果可以看出来:上面是调整法建堆并排序所耗时间,下面是插入法建堆并排序所耗时间。基本符合上述分析。




这篇关于用调整法和插入法建堆的Python实现,不同建堆方式对堆排序性能的影响的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Vue3 的 shallowRef 和 shallowReactive:优化性能

大家对 Vue3 的 ref 和 reactive 都很熟悉,那么对 shallowRef 和 shallowReactive 是否了解呢? 在编程和数据结构中,“shallow”(浅层)通常指对数据结构的最外层进行操作,而不递归地处理其内部或嵌套的数据。这种处理方式关注的是数据结构的第一层属性或元素,而忽略更深层次的嵌套内容。 1. 浅层与深层的对比 1.1 浅层(Shallow) 定义

性能测试介绍

性能测试是一种测试方法,旨在评估系统、应用程序或组件在现实场景中的性能表现和可靠性。它通常用于衡量系统在不同负载条件下的响应时间、吞吐量、资源利用率、稳定性和可扩展性等关键指标。 为什么要进行性能测试 通过性能测试,可以确定系统是否能够满足预期的性能要求,找出性能瓶颈和潜在的问题,并进行优化和调整。 发现性能瓶颈:性能测试可以帮助发现系统的性能瓶颈,即系统在高负载或高并发情况下可能出现的问题

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

2. c#从不同cs的文件调用函数

1.文件目录如下: 2. Program.cs文件的主函数如下 using System;using System.Collections.Generic;using System.Linq;using System.Threading.Tasks;using System.Windows.Forms;namespace datasAnalysis{internal static

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal