10、【堆】左倾堆

一、左倾堆的介绍

左倾堆(leftist tree 或 leftist heap),又被成为左偏树、左偏堆,最左堆等。
它和二叉堆一样,都是优先队列实现方式。当优先队列中涉及到"对两个优先队列进行合并"的问题时,二叉堆的效率就无法令人满意了,而本文介绍的左倾堆,则可以很好地解决这类问题。

左倾堆的定义

左倾堆是一棵二叉树,它的节点除了和二叉树的节点一样具有左右子树指针外,还有两个属性:键值零距离
(01) 键值的作用是来比较节点的大小,从而对节点进行排序。
(02) 零距离(英文名NPL,即Null Path Length)则是从一个节点到一个"最近的不满节点"的路径长度。不满节点是指该该节点的左右孩子至少有有一个为NULL。叶节点的NPL为0,NULL节点的NPL为-1。

左倾堆的基本性质:
[性质1] 节点的键值小于或等于它的左右子节点的键值。
[性质2] 节点的左孩子的NPL >= 右孩子的NPL。
[性质3] 节点的NPL = 它的右孩子的NPL + 1。

左倾堆,顾名思义,是有点向左倾斜的意思了。它在统计问题、最值问题、模拟问题和贪心问题等问题中有着广泛的应用。此外,斜堆是比左倾堆更为一般的数据结构。当然,今天讨论的是左倾堆,关于斜堆,以后再撰文来表。
前面说过,它能和好的解决"两个优先队列合并"的问题。实际上,左倾堆的合并操作的平摊时间复杂度为O(lg

n),而完全二叉堆为O(n)。合并就是左倾树的重点,插入和删除操作都是以合并操作为基础的。插入操作,可以看作两颗左倾树合并;删除操作(移除优先队列中队首元素),则是移除根节点之后再合并剩余的两个左倾树。

二、左倾堆解析

1. 左倾堆基本定义

 1 template <class T>
 2 class LeftistNode{
 3     public:
 4         T key;                // 关键字(键值)
 5         int npl;            // 零路经长度(Null Path Length)
 6         LeftistNode *left;    // 左孩子
 7         LeftistNode *right;    // 右孩子
 8
 9         LeftistNode(T value, LeftistNode *l, LeftistNode *r):
10             key(value), npl(0), left(l),right(r) {}
11 };

LeftistNode是左倾堆对应的节点类。

 1 template <class T>
 2 class LeftistHeap {
 3     private:
 4         LeftistNode<T> *mRoot;    // 根结点
 5
 6     public:
 7         LeftistHeap();
 8         ~LeftistHeap();
 9
10         // 前序遍历"左倾堆"
11         void preOrder();
12         // 中序遍历"左倾堆"
13         void inOrder();
14         // 后序遍历"左倾堆"
15         void postOrder();
16
17          // 将other的左倾堆合并到this中。
18         void merge(LeftistHeap<T>* other);
19         // 将结点(key为节点键值)插入到左倾堆中
20         void insert(T key);
21         // 删除结点(key为节点键值)
22         void remove();
23
24         // 销毁左倾堆
25         void destroy();
26
27         // 打印左倾堆
28         void print();
29     private:
30
31         // 前序遍历"左倾堆"
32         void preOrder(LeftistNode<T>* heap) const;
33         // 中序遍历"左倾堆"
34         void inOrder(LeftistNode<T>* heap) const;
35         // 后序遍历"左倾堆"
36         void postOrder(LeftistNode<T>* heap) const;
37
38         // 交换节点x和节点y
39         void swapNode(LeftistNode<T> *&x, LeftistNode<T> *&y);
40         // 合并"左倾堆x"和"左倾堆y"
41         LeftistNode<T>* merge(LeftistNode<T>* &x, LeftistNode<T>* &y);
42         // 将结点(z)插入到左倾堆(heap)中
43         LeftistNode<T>* insert(LeftistNode<T>* &heap, T key);
44         // 删除左倾堆(heap)中的结点(z),并返回被删除的结点
45         LeftistNode<T>* remove(LeftistNode<T>* &heap);
46
47         // 销毁左倾堆
48         void destroy(LeftistNode<T>* &heap);
49
50         // 打印左倾堆
51         void print(LeftistNode<T>* heap, T key, int direction);
52 };

LeftistHeap是左倾堆类,它包含了左倾堆的根节点,以及左倾堆的操作。

2. 合并

合并操作是左倾堆的重点。合并两个左倾堆的基本思想如下:
  (1) 如果一个空左倾堆与一个非空左倾堆合并,返回非空左倾堆。
  (2) 如果两个左倾堆都非空,那么比较两个根节点,取较小堆的根节点为新的根节点。将"较小堆的根节点的右孩子"和"较大堆"进行合并。
  (3) 如果新堆的右孩子的NPL > 左孩子的NPL,则交换左右孩子。
  (4) 设置新堆的根节点的NPL = 右子堆NPL + 1

 1 /*
 2  * 合并"左倾堆x"和"左倾堆y"
 3  */
 4 template <class T>
 5 LeftistNode<T>* LeftistHeap<T>::merge(LeftistNode<T>* &x, LeftistNode<T>* &y)
 6 {
 7     if(x == NULL)
 8         return y;
 9     if(y == NULL)
10         return x;
11
12     // 合并x和y时,将x作为合并后的树的根;
13     // 这里的操作是保证: x的key < y的key
14     if(x->key > y->key)
15         swapNode(x, y);
16
17     // 将x的右孩子和y合并,"合并后的树的根"是x的右孩子。
18     x->right = merge(x->right, y);
19
20     // 如果"x的左孩子为空" 或者 "x的左孩子的npl<右孩子的npl"
21     // 则,交换x和y
22     if(x->left == NULL || x->left->npl < x->right->npl)
23     {
24         LeftistNode<T> *tmp = x->left;
25         x->left = x->right;
26         x->right = tmp;
27     }
28     // 设置合并后的新树(x)的npl
29     if (x->right == NULL || x->left == NULL)
30         x->npl = 0;
31     else
32         x->npl = (x->left->npl > x->right->npl) ? (x->right->npl + 1) : (x->left->npl + 1);
33
34     return x;
35 }
36
37 /*
38  * 将other的左倾堆合并到this中。
39  */
40 template <class T>
41 void LeftistHeap<T>::merge(LeftistHeap<T>* other)
42 {
43     mRoot = merge(mRoot, other->mRoot);
44 }

merge(x, y)是内部接口,作用是合并x和y这两个左倾堆,并返回得到的新堆的根节点。
merge(other)是外部接口,作用是将other合并到当前堆中。

3. 添加

 1 /*
 2  * 将结点插入到左倾堆中,并返回根节点
 3  *
 4  * 参数说明:
 5  *     heap 左倾堆的根结点
 6  *     key 插入的结点的键值
 7  * 返回值:
 8  *     根节点
 9  */
10 template <class T>
11 LeftistNode<T>* LeftistHeap<T>::insert(LeftistNode<T>* &heap, T key)
12 {
13     LeftistNode<T> *node;    // 新建结点
14
15     // 新建节点
16     node = new LeftistNode<T>(key, NULL, NULL);
17     if (node==NULL)
18     {
19         cout << "ERROR: create node failed!" << endl;
20         return heap;
21     }
22
23     return merge(mRoot, node);
24 }
25
26 template <class T>
27 void LeftistHeap<T>::insert(T key)
28 {
29     mRoot = insert(mRoot, key);
30 }

insert(heap, key)是内部接口,它是以节点为操作对象的。
insert(key)是外部接口,它的作用是新建键值为key的节点,并将其加入到当前左倾堆中。

4. 删除

 1 /*
 2  * 删除结点,返回根节点
 3  *
 4  * 参数说明:
 5  *     heap 左倾堆的根结点
 6  * 返回值:
 7  *     根节点
 8  */
 9 template <class T>
10 LeftistNode<T>* LeftistHeap<T>::remove(LeftistNode<T>* &heap)
11 {
12     if (heap == NULL)
13         return NULL;
14
15     LeftistNode<T> *l = heap->left;
16     LeftistNode<T> *r = heap->right;
17
18     // 删除根节点
19     delete heap;
20
21     return merge(l, r); // 返回左右子树合并后的新树
22 }
23
24 template <class T>
25 void LeftistHeap<T>::remove()
26 {
27     mRoot = remove(mRoot);
28 }

remove(heap)是内部接口,它是以节点为操作对象的。
remove()是外部接口,它的作用是删除左倾堆的最小节点。

原文地址:https://www.cnblogs.com/Long-w/p/9786173.html

时间: 2024-11-13 09:42:38

10、【堆】左倾堆的相关文章

左倾堆

左倾堆(或左偏树)和之前记录过的二叉堆一样,是堆的一种:和普通的二叉堆不同,它是一种可合并堆.可合并堆相比于普通的二叉堆在对两个堆进行合并的操作上具有很大的优势:对于基本的二叉堆合并,时间复杂度为O(n), 而对于可合并堆,其时间复杂度为O(log2n). 左倾堆性质 左倾堆(也叫左偏树),是一种可合并堆.它有以下性质: 每个节点含有左子结点指针.右子节点指针.键值key.NPL(Null Path Length,表示当前节点到最近的一个不满子节点的长度,不满节点指的是节点含有少于两个子节点)

10/16 对顶堆算法研究(POJ 3784)

/* 考虑维护两个堆 一个堆是大根堆,存储1-x的元素 一个堆是小根堆,存储x+1-N的元素 对于一个加入的元素y,考虑将其加入大根堆or小根堆? 如果y>mid,那么将其加入小根堆(上面的堆) 如果y<mid,那么将其加入大根堆(下面的堆) 并且在每一次操作之后维护堆的状态是合法的! 也就是维护堆得大小正确 */#include<cstdio>#include<algorithm>#include<cstring>#include<queue>

数据结构之二叉堆(构建堆,堆排序)-(七)

/* * @author Lip * @date 2015-04-23 */ public class Heap { public static void main(String[] args) { // TODO Auto-generated method stub //System.out.println((int)-1.5); int []array={4,2,7,9,3,6,1,12,10,5}; System.out.println("原始:"); printHeapByLe

C++ 堆 和 堆 分析

[摘要] 堆和栈,即是数据结构,又是分配存储空间的不同方式.在数据结构上.堆是树型层次结构,结点按keyword次序排列,经常使用的堆为二叉堆:栈是一种先进后出的数据结构.在内存分配上的堆和栈,首要差别在于申请方式不同.其次在存取速度.存储空间的大小.存储内容(一定要记住,栈中是第一条可运行语句地址.然后是各个參数.堆中头部是堆的大小描写叙述.之后有程序猿自己安排).内存中的相对位置和系统相应的响应上都各有自己差别.在C语言 的学习过程中,堆和栈即是基础也是重点. [正文] 堆栈是一个非常模糊的

堆是堆,栈归栈

堆是堆,栈归栈 在阅读以下内容之前,请了解一下几点: 第一:坚决澄清:堆是堆,栈归栈. 第二:曾经的“堆栈”再不允许重谈,简直就是扯淡! 第三:下面内容均属于从内存分配角度的阐述,不要与数据结构混淆. [1]程序的内存分配 (1)内存分配详解 一个由C/C++编译的程序占用的内存分为以下几个部分 <1>栈区(stack)— 由编译器自动分配释放,存放函数的参数值,局部变量的值等. <2>堆区(heap) — 一般由程序员设计分配及释放,若程序员不释放,程序结束时可能由OS回收.可能

bzoj 1577: [Usaco2009 Feb]庙会捷运Fair Shuttle——小根堆+大根堆+贪心

Description 公交车一共经过N(1<=N<=20000)个站点,从站点1一直驶到站点N.K(1<=K<=50000)群奶牛希望搭乘这辆公交车.第i群牛一共有Mi(1<=Mi<=N)只. 他们希望从Si到Ei去.公交车只能座C(1<=C<=100)只奶牛.而且不走重复路线,请计算这辆车最多能满足多少奶牛听要求.注意:对于每一群奶牛,可以部分满足,也可以全部满足,也可以全部不满足. Input 第1行: 三个整数: K,N,C. 由空格隔开. 第2..

堆及堆的变种

堆及堆的变种 声明 参考课件和讲授来自Accelerator,分析懒得打也来自他 堆的元素删除 借用标记的思想,我们维护一个和原堆同样性质(大根,小根)的堆,每次删除就把它扔到标记堆里面 当我们需要 pop 的时候,如果堆顶元素和删除堆顶元素相同, 那么就说明这个元素是我们之前删除过的,于是我们就在删除堆里面和这个堆里面同时 pop, 然后考察下一元素. 容易发现时间复杂度和空间复杂度没有什么实质性的变化. 题 luogu P3545 [POI2012]HUR-Warehouse Store 我

SQL0973N在 &quot;&lt;堆名&gt;&quot; 堆中没有足够的存储器可用来处理语句

SQL0973N在 "<堆名>" 堆中没有足够的存储器可用来处理语句. 解释: 已使用此堆的所有可用内存.不能处理该语句. 用户响应: 接收到此消息(SQLCODE)后就终止应用程序.修改 "<堆名称>"配置参数以增大堆大小. 例如,要更新数据库配置参数,发出如下命令: db2 update db cfg  for "<db-name>"  using "<heap-name>"

C#中的堆栈与堆(托管堆) [转自互联网]

首先堆栈和堆(托管堆)都在进程的虚拟内存中.(在32位处理器上每个进程的虚拟内存为4GB)堆栈stack : 堆栈中存储值类型.   堆栈实际上是向下填充,即由高内存地址指向地内存地址填充.   堆栈的工作方式是先分配内存的变量后释放(先进后出原则).   堆栈中的变量是从下向上释放,这样就保证了堆栈中先进后出的规则不与变量的生命周期起冲突!   堆栈的性能非常高,但是对于所有的变量来说还不太灵活,而且变量的生命周期必须嵌套. 通常我们希望使用一种方法分配内存来存储数据,并且方法退出后很长一段时