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

相关文章

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、

Springboot处理跨域的实现方式(附Demo)

《Springboot处理跨域的实现方式(附Demo)》:本文主要介绍Springboot处理跨域的实现方式(附Demo),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不... 目录Springboot处理跨域的方式1. 基本知识2. @CrossOrigin3. 全局跨域设置4.

springboot security使用jwt认证方式

《springbootsecurity使用jwt认证方式》:本文主要介绍springbootsecurity使用jwt认证方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录前言代码示例依赖定义mapper定义用户信息的实体beansecurity相关的类提供登录接口测试提供一

Spring Boot 3.4.3 基于 Spring WebFlux 实现 SSE 功能(代码示例)

《SpringBoot3.4.3基于SpringWebFlux实现SSE功能(代码示例)》SpringBoot3.4.3结合SpringWebFlux实现SSE功能,为实时数据推送提供... 目录1. SSE 简介1.1 什么是 SSE?1.2 SSE 的优点1.3 适用场景2. Spring WebFlu

基于SpringBoot实现文件秒传功能

《基于SpringBoot实现文件秒传功能》在开发Web应用时,文件上传是一个常见需求,然而,当用户需要上传大文件或相同文件多次时,会造成带宽浪费和服务器存储冗余,此时可以使用文件秒传技术通过识别重复... 目录前言文件秒传原理代码实现1. 创建项目基础结构2. 创建上传存储代码3. 创建Result类4.

SpringBoot日志配置SLF4J和Logback的方法实现

《SpringBoot日志配置SLF4J和Logback的方法实现》日志记录是不可或缺的一部分,本文主要介绍了SpringBoot日志配置SLF4J和Logback的方法实现,文中通过示例代码介绍的非... 目录一、前言二、案例一:初识日志三、案例二:使用Lombok输出日志四、案例三:配置Logback一

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.