计算机程序的思维逻辑 (45) - 神奇的堆
|
在队列中,一般是从头部删除元素,Java中用堆实现优先级队列,我们来看下如何在堆中删除头部,其基本步骤为:
我们来看个例子。下面是初始结构:
从中间删除元素 那如果需要从中间删除某个节点呢?与从头部删除一样,都是先用最后一个元素替换待删元素。不过替换后,有两种情况,如果该元素大于某孩子节点,则需向下调整(siftdown),否则,如果小于父节点,则需向上调整(siftup)。 我们来看个例子,删除值为21的节点,第一步如下图所示:
给定一个无序数组,如何使之成为一个最小堆呢?将普通无序数组变为堆的过程我们称之为heapify。 基本思路是,从最后一个非叶子节点开始,一直往前直到根,对每个节点,执行向下调整siftdown。换句话说,是自底向上,先使每个最小子树为堆,然后每对左右子树和其父节点合并,调整为更大的堆,因为每个子树已经为堆,所以调整就是对父节点执行siftdown,就这样一直合并调整直到根。这个算法的伪代码是: void heapify() {
for (int i=size/2; i >= 1; i--)
siftdown(i);
}
size表示节点个数, 节点编号从1开始,size/2表示第一个非叶节点的编号。 这个构建的时间效率为O(N),N为节点个数,具体就不证明了。 查找和遍历 在堆中进行查找没有特殊的算法,就是从数组的头找到尾,效率为O(N)。 在堆中进行遍历也是类似的,堆就是数组,堆的遍历就是数组的遍历,第一个元素是最大值或最小值,但后面的元素没有特定的顺序。 需要说明的是,如果是逐个从头部删除元素,堆可以确保输出是有序的。 算法小结 (编辑:网站开发网_安阳站长网) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |









