【编程珠玑】第十四章 堆(排序,优先级队列)

2024-04-05 01:48

本文主要是介绍【编程珠玑】第十四章 堆(排序,优先级队列),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一,堆

       1)堆:任何结点的值都小于或等于其孩子的值的完全二叉树为小根堆

                    任何结点的值都大于或等于其孩子的值的完全二叉树为大根堆

      为了方便使用完全二叉树的性质,数组从下标1开始。

            这样:leftChild = 2*i ;  

                       rightChild = 2*i + 1 ;  

                       parent = i/2 ; 

                       null   i < 1  or  i > n


       2)堆算法分析

            堆排序的最坏时间复杂度为O(nlogn)。堆序的平均性能较接近于最坏性能。

       由于建初始堆所需的比较次数较多,所以堆排序不适宜于记录数较少的文件。
       堆排序是就地排序,辅助空间为O(1)

            堆排序是不稳定的

  

       3)堆实现

            【特别注意】堆不一定是完全二叉树但是一般采用完全二叉树,主要是利于存储和运算。堆排序作用在数组上

                                  

             初始建立堆:

                        给一个数组,将数组看做完全二叉树。

                        从最后一个非叶结点(length/2,下标从1开始),直到第一个结点a[1],向上调整建立堆。

             排序和堆调整

                        将第一个值a[1] 跟最后一个值交换,然后对 a[1] 调整堆(此时数组长度调整为length-1)


            【注意】这里初始建堆,只考虑已经有n个元素,向下调整建堆就可以搞定。

                           但是对于insert(t)怎么办? 采用向上调整堆的策略。详细见下文优先队列。

       4)源码

#include"stdio.h"
inline void swap(int &a,int &b)
{
int temp=a;
a=b;
b=temp;
}
void HeapAdjust(int array[],int i,int nLength)//自顶向下调整堆
{
int nChild;
int nTemp;//赋值为待调整的 节点
for(nTemp=array[i];2*i<nLength;i=nChild)//2*i<nLength说明还有左孩子
{
nChild=2*i;//左孩子  
/*一共两个子节点的话得到 较大的一个*/		   
if(nChild<nLength-1&&array[nChild+1]>array[nChild])//nChild<nLength-1 判断到头没有
++nChild;
/*如果较大子节点大于父节点  将子节点 调整到父节点*/
if(nTemp<array[nChild])
array[i]=array[nChild];
else
break;//这个地方不加 会出错  第一个会输出第二个 
array[nChild]=nTemp;//子节点 等于父节点 (不执行break)
} 
}
void HeapSort(int a[],int length)
{
/*初建堆 */
for(int i=length/2;i>0;--i)//从最后一个 非叶子节点调整 (这里的  i是下标) 
HeapAdjust(a,i,length);
for(int i=length;i>1;--i)
{
swap(a[1],a[i]);	/*第一个最大元素跟最后一个交换*/
HeapAdjust(a,1,i);//调整堆 (注意 length=i  由于堆是逐渐变小的)
}
}
int main()
{
int a[10]={0,1,2,5,3,8,4,7,6};
HeapSort(a,8);
for(int i=1;i<9;i++)
printf("%d\n",a[i]);
return 0;
}


二,优先队列

        1)优先队列是0个或多个元素的集合,每个元素都有一个优先权或值,对优先队列执行的操作有1) 查找; 2) 插入一个新元素; 3) 删除.

              在最小优先队列(min priorityq u e u e)中,查找操作用来搜索优先权最小的元素,删除操作用来删除该元素;

              对于最大优先队列(max priority queue),查找操作用来搜索优先权最大的元素,删除操作用来删除该元素.

              优先权队列中的元素可以有相同的优先权,查找与删除操作可根据任意优先权进行.

            

        2)优先队列实现

              初始化一个数组,向空数组依次插入元素,每插入一个元素向上调整一次堆。

              删除元素,将第一个元素跟最后一个元素交换,并向下调整堆

   

        3)代码实现

#include <iostream>
using namespace std;
template<class T>
class priqueue {
private:
int	n, maxsize;
T	*x;
void swap(int &i, int &j)//根据坐标交换数组元素的值
{	T t = i; i = j; j = t; }
public:
priqueue(int m)//初始化数组
{	maxsize = m;
x = new T[maxsize+1];
n = 0;
}
void insert(T t)
{	int i, p;
x[++n] = t; //插入的元素放到最后
for (i = n; i > 1 && x[p=i/2] > x[i]; i = p)    
swap(x[p], x[i]);
}
T extractmin()//向下调整堆
{	           
int i, c;
T t = x[1];
x[1] = x[n--];
for (i = 1; (c=2*i) <= n; i = c) {
if (c+1<=n && x[c+1]<x[c])
c++;
if (x[i] <= x[c])
break;
swap(x[c], x[i]);
}
return t;
}
void print(int n)
{
for (int i = 1; i < n; i++) //输出堆
cout << x[i] << " ";
}
};
template<class T>
void pqsort(T v[], int n)//先初始化一个数组,然后插入建立一个堆
{	priqueue<T> pq(n); 
int i;
for (i = 0; i < n; i++)
pq.insert(v[i]);
cout<<"输出排序后的堆:";
pq.print(n);
}
int main()
{	const int	n = 10;
int	i, v[n];
/*以下是通过向上调整堆 建立一个10个元素的堆*/
for (i = 0; i < n; i++)
v[i] = n-i;      
pqsort(v, n);
cout<<"\n执行插入和删除操作(输入0代表删除最小值,输入其他代表插入)"<<endl;
priqueue<int> pq(100);
int count=0;
while (cin >> i)
if (i == 0)
{
if(count)
cout <<"删除的最小元素为:"<<pq.extractmin() << "\n";
else
cout<<"请先插入元素"<<endl;
}
else
{
pq.insert(i);
count++;
}
return 0;
}


三,习题

       1)为了提高向上调整堆的速度,在x[0] 放置哨兵=当前插入的元素。省去了每次都判断  i>1

             向上调整堆结束:x[p] <= x[i]  

 

void insert(T t) //向上调整堆
{	int i, p;
x[++n] = t; //插入的元素放到最后
x[0]=t;
for (i = n;  x[p=i/2] > x[i]; i = p)    
swap(x[p], x[i]);
}

        4)a.构造哈夫曼树时候,需要选取当前数组的两个最小值,删除两个最小值,并将计算之和插入原来数组。

                 采用堆,初建堆,两次调用选取最小值的函数。计算之和之后,调用插入堆并调整堆


             b.如果将较小浮点数和较大浮点数相加可能造成丢失精度。所以每次取最小的两个相加。然后将和插入数组集合。最后剩下一个就是所有浮点数的和


             c.典型的topK


             d.将所有小文件 要插入的当前值组成一个堆。

                 取堆最小值,插入排序数组。调整堆。然后插入该小文件下一个元素(无后继则不操作)


        5)剩余容量组成堆,权值升序插入堆


        6)求指教(没看懂)


        7)求指教(没看懂)



这篇关于【编程珠玑】第十四章 堆(排序,优先级队列)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

关于Java内存访问重排序的研究

《关于Java内存访问重排序的研究》文章主要介绍了重排序现象及其在多线程编程中的影响,包括内存可见性问题和Java内存模型中对重排序的规则... 目录什么是重排序重排序图解重排序实验as-if-serial语义内存访问重排序与内存可见性内存访问重排序与Java内存模型重排序示意表内存屏障内存屏障示意表Int

C#反射编程之GetConstructor()方法解读

《C#反射编程之GetConstructor()方法解读》C#中Type类的GetConstructor()方法用于获取指定类型的构造函数,该方法有多个重载版本,可以根据不同的参数获取不同特性的构造函... 目录C# GetConstructor()方法有4个重载以GetConstructor(Type[]

hdu1180(广搜+优先队列)

此题要求最少到达目标点T的最短时间,所以我选择了广度优先搜索,并且要用到优先队列。 另外此题注意点较多,比如说可以在某个点停留,我wa了好多两次,就是因为忽略了这一点,然后参考了大神的思想,然后经过反复修改才AC的 这是我的代码 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

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

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

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

hdu 1285(拓扑排序)

题意: 给各个队间的胜负关系,让排名次,名词相同按从小到大排。 解析: 拓扑排序是应用于有向无回路图(Direct Acyclic Graph,简称DAG)上的一种排序方式,对一个有向无回路图进行拓扑排序后,所有的顶点形成一个序列,对所有边(u,v),满足u 在v 的前面。该序列说明了顶点表示的事件或状态发生的整体顺序。比较经典的是在工程活动上,某些工程完成后,另一些工程才能继续,此时

poj 3190 优先队列+贪心

题意: 有n头牛,分别给他们挤奶的时间。 然后每头牛挤奶的时候都要在一个stall里面,并且每个stall每次只能占用一头牛。 问最少需要多少个stall,并输出每头牛所在的stall。 e.g 样例: INPUT: 51 102 43 65 84 7 OUTPUT: 412324 HINT: Explanation of the s

poj 2431 poj 3253 优先队列的运用

poj 2431: 题意: 一条路起点为0, 终点为l。 卡车初始时在0点,并且有p升油,假设油箱无限大。 给n个加油站,每个加油站距离终点 l 距离为 x[i],可以加的油量为fuel[i]。 问最少加几次油可以到达终点,若不能到达,输出-1。 解析: 《挑战程序设计竞赛》: “在卡车开往终点的途中,只有在加油站才可以加油。但是,如果认为“在到达加油站i时,就获得了一