堆结构

堆是一种特殊的树,它需要满足以下两点:

对于每个节点的值都大于等于子树中每个节点值的堆,我们叫作大顶堆。对于每个节点的值都小于等于子树中每个节点值的堆,我们叫作小顶堆

1,2是大顶堆,3是小顶堆,4不是堆。

1,2是大顶堆,3是小顶堆,4不是堆。

在使用堆排序的时候,如果有添加删除元素,都要调整以满足堆的两个特性。

对于新加入元素,我们将其放到数组最后(因为堆是一个完全二叉树,可以用数组存储),然后自下而上依次比较并交换

插入元素

插入元素

删除堆顶元素

删除堆顶元素

我们把最后一个节点放到堆顶,然后利用同样的父子节点对比方法。对于不满足父子节点大小关系的,互换两个节点,并且重复进行这个过程,直到父子节点之间满足大小关系为止。这就是从上往下的堆化方法

因为我们移除的是数组中的最后一个元素,而在堆化的过程中,都是交换操作,不会出现数组中的“空洞”,所以这种方法堆化之后的结果,肯定满足完全二叉树的特性。

由于一个包含 n 个节点的完全二叉树,树的高度不会超过 log2n。堆化的过程是顺着节点所在路径比较交换的,所以堆化的时间复杂度跟树的高度成正比,也就是 O(logn)。插入数据和删除堆顶元素的主要逻辑就是堆化,所以,往堆中插入一个元素和删除堆顶元素的时间复杂度都是 O(logn)。

public class Heap {
  private int[] a; // 数组,从下标 1 开始存储数据
  private int n;  // 堆可以存储的最大数据个数
  private int count; // 堆中已经存储的数据个数
 
  public Heap(int capacity) {
    a = new int[capacity + 1];
    n = capacity;
    count = 0;
  }
 
  public void insert(int data) {
    if (count >= n) return; // 堆满了
    ++count;
    a[count] = data;
    int i = count;
    while (i/2 > 0 && a[i] > a[i/2]) { // 自下往上堆化
      swap(a, i, i/2); // swap() 函数作用:交换下标为 i 和 i/2 的两个元素
      i = i/2;
    }
  }

	public void removeMax() {
	  if (count == 0) return -1; // 堆中没有数据
	  a[1] = a[count];
	  --count;
	  heapify(a, count, 1);
	}
	 
	private void heapify(int[] a, int n, int i) { // 自上往下堆化
	  while (true) {
	    int maxPos = i;
	    if (i*2 <= n && a[i] < a[i*2]) maxPos = i*2;
	    if (i*2+1 <= n && a[maxPos] < a[i*2+1]) maxPos = i*2+1;
	    if (maxPos == i) break;
	    swap(a, i, maxPos);
	    i = maxPos;
	  }
	}
 }

在上面的代码中,都是假设堆中的数据是从数组下标为 1 的位置开始存储。那如果从 0 开始存储,实际上处理思路是没有任何变化的,唯一变化的,可能就是,代码实现的时候,计算子节点和父节点的下标的公式改变了。

如果节点的下标是 i,那左子节点的下标就是 $2 * i + 1$,右子节点的下标就是 $2 * i + 2$,父节点的下标就是 $(i-1)/2$。

<aside> 💡 用数组存储表示完全二叉树时,也可以从下标为0开始,只是这样做的话,计算左子节点时,会多一次加法运算

</aside>

堆排序

我们借助于堆这种数据结构实现的排序算法,就叫作堆排序。这种排序方法的时间复杂度非常稳定,是 O(nlogn),并且它还是原地排序算法。堆排序的过程可以大致分解成两个大的步骤,建堆排序

建堆

建堆的过程,有两种思路。

第一种是借助我们前面讲的,在堆中插入一个元素的思路。尽管数组中包含 n 个数据,但是我们可以假设,起初堆中只包含一个数据,就是下标为 1 的数据。然后,我们调用前面讲的插入操作,将下标从 2 到 n 的数据依次插入到堆中。这样我们就将包含 n 个数据的数组,组织成了堆。

第二种实现思路,跟第一种截然相反。第一种建堆思路的处理过程是从前往后处理数组数据,并且每个数据插入堆中时,都是从下往上堆化。而第二种实现思路,是从后往前处理数组,并且每个数据都是从上往下堆化。

private static void buildHeap(int[] a, int n) {
	// 对于完全二叉树来说,下标从 n/2+1 到 n 的节点都是叶子节点,所以只需要从 n/2 开始即可
  for (int i = n/2; i >= 1; --i) {
    heapify(a, n, i);
  }
}
 
private static void heapify(int[] a, int n, int i) {
  while (true) {
    int maxPos = i;
    if (i*2 <= n && a[i] < a[i*2]) maxPos = i*2;
    if (i*2+1 <= n && a[maxPos] < a[i*2+1]) maxPos = i*2+1;
    if (maxPos == i) break;
    swap(a, i, maxPos);
    i = maxPos;
  }
}

现在,我们来看,建堆操作的时间复杂度是多少呢?