用调整法和插入法建堆的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

相关文章

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

SpringBoot中@Value注入静态变量方式

《SpringBoot中@Value注入静态变量方式》SpringBoot中静态变量无法直接用@Value注入,需通过setter方法,@Value(${})从属性文件获取值,@Value(#{})用... 目录项目场景解决方案注解说明1、@Value("${}")使用示例2、@Value("#{}"php

SpringBoot分段处理List集合多线程批量插入数据方式

《SpringBoot分段处理List集合多线程批量插入数据方式》文章介绍如何处理大数据量List批量插入数据库的优化方案:通过拆分List并分配独立线程处理,结合Spring线程池与异步方法提升效率... 目录项目场景解决方案1.实体类2.Mapper3.spring容器注入线程池bejsan对象4.创建

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python正则表达式匹配和替换的操作指南

《Python正则表达式匹配和替换的操作指南》正则表达式是处理文本的强大工具,Python通过re模块提供了完整的正则表达式功能,本文将通过代码示例详细介绍Python中的正则匹配和替换操作,需要的朋... 目录基础语法导入re模块基本元字符常用匹配方法1. re.match() - 从字符串开头匹配2.

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

通过Docker容器部署Python环境的全流程

《通过Docker容器部署Python环境的全流程》在现代化开发流程中,Docker因其轻量化、环境隔离和跨平台一致性的特性,已成为部署Python应用的标准工具,本文将详细演示如何通过Docker容... 目录引言一、docker与python的协同优势二、核心步骤详解三、进阶配置技巧四、生产环境最佳实践

Python一次性将指定版本所有包上传PyPI镜像解决方案

《Python一次性将指定版本所有包上传PyPI镜像解决方案》本文主要介绍了一个安全、完整、可离线部署的解决方案,用于一次性准备指定Python版本的所有包,然后导出到内网环境,感兴趣的小伙伴可以跟随... 目录为什么需要这个方案完整解决方案1. 项目目录结构2. 创建智能下载脚本3. 创建包清单生成脚本4

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF