6.1堆

卫星数据:一个带排序的的数通常是有一个称为记录的数据集组成的,每一个记录有一个关键字key,记录的其他数据称为卫星数据。

原地排序:在排序输入数组时,只有常数个元素被存放到数组以外的空间中去。

在第二章介绍了两种排序:插入排序和合并排序,接下来两章要介绍的是推排序和快速排序,这四个排序都属于比较排序(comparison sort)。

快速排序的性能一般优先于堆排序

二叉堆是一个数组(b),近似完全二叉树(a)

数组(b) 实际的存储形势

二叉树(a) 要表达的结构

[1,A.heap-size] 闭区间的范围是堆(python一0开始 选择 [0,A.heap-size+1]的数组 怎么舍弃第一位? -1 ? 0 ? )

PARENT 返回的i的父亲结点在数组b中的下标 如i = 2 ->PARENT(2) ==1

LEFT 返回的i的左孩子结点在数组b中的下标 如i = 2 ->LEFT(2) ==4

RIGHT 返回的i的右孩子结点在数组b中的下标 如i = 2 ->LEFT(2) ==5

这3条是用数组(顺序表)表达堆(完全二叉树)的关键

特别注明i(数组下标)必须是从1开始{python中我舍弃第一位 data = [-1,] }

def PARENT(i):

    return i//2 #为什么是一半 参考离散数学和数据结构

                        #我的解释是:二叉树的性质+下标从1开始

def LEFT(i):

    return i*2 #同上

def RIGHT(i):

    return i*2 + 1 #同上

最大堆定义:所以父结点比孩子结点大

#堆排序使用 关键所在 调堆和建堆都是保证他

#堆排序就是将他的根节点{堆顶}取出来-->调堆-->将他的根节点{堆顶}取出来-->调堆-...直到堆只有一个数

最小堆定义:所以父结点比孩子结点小

6.2维护堆的性质(调堆)

讲A[i] 和他的2个孩子对比最大的放在A[i]的位置

如果A[i]就是最大值的就结束

否则交换A[i] 和A[x] 再递归调节x结点

上图:

参考

class Mylist(list):

    def __init__(self):

        self.heap_size = 0

        super().__init__()

def MAX_HEAPIFY(A,i):

    l = LEFT(i)

    r = RIGHT(i)

    #找出最大的结点

    #i的左孩子是否大于i

    #A.heap_size 写一个继承了list类 类中加上这个参数(Mylist)

    #或者选择A[0] 位放heap_size ??

    #或者设计全局变量

    if l <= A.heap_size and A[l] > A[i]:

        largest = l

    else:

        largest = i

    #和右孩子比

    if r <= A.heap_size and A[r] > A[largest]:

        largest = r

    if largest != i: #如果A[i]不是最大的 就要调堆了

        A[i],A[largest] = A[largest],A[i] #交换

        MAX_HEAPIFY(A,largest) #递归调largest

6.3建堆

从最后一个有子树的(图中坐标5 = 10//2)的结点向根节点(图中坐标1)调堆

def BUILD_MAX_HEAP(A):

    A.heap_size = len(A)-1

    #print(len(A))

    for i in range(A.heap_size//2,0,-1): #从n//2开始到1

        #print(i)

        MAX_HEAPIFY(A,i)

(在纸上画一遍就知道怎么回事了)

O(n)

6.4堆排序算法

建立最大堆后(如a)显然16(A[1])是最大的,取出最大的,再调堆,再取出最大的。。。。。

 

def HEAPSORT(A):

    BUILD_MAX_HEAP(A) #建堆

    print("建成的堆:",A)

    for i in range(len(A)-1,1,-1):

        A[1],A[i] = A[i],A[1] #第一位和最后有位换

        A.heap_size = A.heap_size - 1 #取出了一个

        MAX_HEAPIFY(A,1) #调堆        

if __name__ == '__main__':

    A = Mylist()

    #print(type(A))

    for i in[-1,4,1,3,2,16,9,10,14,8,7]: #A = [,...] A会变成list

        A.append(i)

    #print(type(A))

    HEAPSORT(A)

    print("堆排序后:",A)

'''

============== RESTART: F:\python\algorithms\6_2_max_heapify.py ==============

建成的堆: [-1, 16, 14, 10, 8, 7, 9, 3, 2, 4, 1]

堆排序后: [-1, 1, 2, 3, 4, 7, 8, 9, 10, 14, 16]

环境win7 + python3.5.1

'''

 

 

引用参考:

http://blog.csdn.net/littlethunder/article/details/23877545

http://www.wutianqi.com/?cat=515&paged=5

http://blog.csdn.net/hanchengxi/article/details/8454754

算法导论 第六章 堆排序(python)的更多相关文章

  1. 算法导论 第六章 2 优先队列(python)

    优先队列:     物理结构: 顺序表(典型的是数组){python用到list}     逻辑结构:似完全二叉树 使用的特点是:动态的排序..排序的元素会增加,减少#和快速排序对比 快速一次排完 增 ...

  2. 算法导论 第六章 思考题6-3 Young氏矩阵

    这题利用二叉堆维持堆性质的办法来维持Young氏矩阵的性质,题目提示中写得很清楚,不过确实容易转不过弯来. a,b两问很简单.直接看c小问: 按照Young氏矩阵的性质,最小值肯定在左上角取得,问题在 ...

  3. 算法导论 第六章 思考题 6-3 d叉堆

    d叉堆的实现相对于二叉堆变化不大,首先看它如何用数组表示. 考虑一个索引从1开始的数组,一个结点i最多可以有d个子结点,编号从id - (d - 2) 到 id + 1. 从而可以知道一个结点i的父结 ...

  4. 《算法导论》第二章demo代码实现(Java版)

    <算法导论>第二章demo代码实现(Java版) 前言 表示晚上心里有些不宁静,所以就写一篇博客,来缓缓.囧 拜读<算法导论>这样的神作,当然要做一些练习啦.除了练习题与思考题 ...

  5. 《算法》第六章部分程序 part 7

    ▶ 书中第六章部分程序,加上自己补充的代码,包括全局最小切分 Stoer-Wagner 算法,最小权值二分图匹配 ● 全局最小切分 Stoer-Wagner 算法 package package01; ...

  6. 《算法》第六章部分程序 part 6

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,包括二分图最大匹配(最小顶点覆盖)的交替路径算法和 HopcroftKarp 算法 ● 二分图最大匹配(最小顶点覆盖)的交替路径算法 package ...

  7. 《算法》第六章部分程序 part 5

    ▶ 书中第六章部分程序,包括在加上自己补充的代码,网络最大流 Ford - Fulkerson 算法,以及用到的流量边类和剩余流量网络类 ● 网络最大流 Ford - Fulkerson 算法 pac ...

  8. 为什么我要放弃javaScript数据结构与算法(第六章)—— 集合

    前面已经学习了数组(列表).栈.队列和链表等顺序数据结构.这一章,我们要学习集合,这是一种不允许值重复的顺序数据结构. 本章可以学习到,如何添加和移除值,如何搜索值是否存在,也可以学习如何进行并集.交 ...

  9. 《算法导论》— Chapter 6 堆排序

    序 本文主要介绍堆排序算法(HeapSort),堆排序像合并排序而不像插入排序,堆排序的运行时间为O(nlgn):像插入排序而不像合并排序,它是一种原地(in place)排序算法.在任何时候,数组中 ...

随机推荐

  1. EditPlus 3:设置自动换行

    打开软件,菜单栏点击Document,再在下拉栏中点击Permanent Settings,然后在弹出的设置框中找到Word Wrap点击,最后在弹出的小框中勾选第一个Enable word wrap ...

  2. 通过 xshell 连接 ubuntu on windows(WSL)

    装上 ubuntu on windows 后,默认要先打开 cmd, 再运行 bash 进入 ubuntu 的 shell. 但是这个shell很难看,配色不好就算了,还存在各种复制粘贴麻烦. 默认没 ...

  3. 跟我一起玩Win32开发(11):使用控件——先来耍一下按钮

    用户通过控件与应用程序交互,在吹牛之前,先介绍一个工具,这是官方的工具,使用它,你可以预览常用控件的外观.样式,以及对控进行操作时接收和发送哪些消息.下载地址如下: http://www.micros ...

  4. POJ1961(kmp中Next数组的性质)

    对于某个位置i,i - Next[i]是循环节长度,i整除(i - Next[i])时是完整的几个循环元. ; int n, kase, Next[maxn]; char ch[maxn]; inli ...

  5. hibernate Day1 案例代码

    1.创建Person类 package com.icss.pojo; public class Person { private int uid; private String uname; priv ...

  6. yum卸载遇到的问题--待解决

    系统版本是 [root@master ~]# uname -a Linux master -.el6.x86_64 # SMP Sun Nov :: EST x86_64 x86_64 x86_64 ...

  7. C#方法拓展

    作用: “扩展方法使您能够向现有类型“添加”方法,而无需创建新的派生类型.重新编译或以其他方式修改原始类型.” 要求: 1.拓展方法必须是在一个非嵌套.非泛型的静态类中定义.2.他至少有一个参数.3. ...

  8. C#结构体和类的区别(转)

    结构体和类的区别:    在做一个项目时,使用了较多的结构体,并且存在一些结构体的嵌套,即某结构体成员集合包含另一个结构体等,总是出现一些奇怪的错误,才终于下决心好好分析一下到底类和结构体有啥不同,虽 ...

  9. websocket 加layim实现在线聊天系统

    实现流程: 1.浏览器连接服务器时保存所有用户id以及对应的唯一session(session用户用户消息推送). 1.1:判断登录用户是否有离线消息(个人消息以及群消息),有则将离线消息进行推送给登 ...

  10. bootstrap CSS表单、按钮和字体图标

    基础表单   <form role="form">     <div class="form-group">         <l ...