二叉堆 堆排序 優先隊列

二叉堆是一个近似完满二叉树的结构,并同时满足堆的性质:
即子结点的键值或索引总是小于(或者大于)它的父节点。

通常堆是通过一维数组来实现的。在起始数组为 0 的情形中:
父节点i的左子节点在位置 (2*i+1);
父节点i的右子节点在位置 (2*i+2);
子节点i的父节点在位置 floor((i-1)/2);

在堆的数据结构中,堆中的最大值总是位于根节点。

堆中定义以下几种操作:
堆调整(heapify):将堆的末端子结点作调整,维持堆的性质
创建堆(build_heap):将一个无序数组构造为一个堆。
堆排序(heap_sort):移除位在第一个数据的根结点,并做堆调整的递归运算

 

#include <stdio.h>
#include <stdlib.h>
#include <math.h>

int parent(int);
int left(int);
int right(int);
void heapify(int [], int, int);
void build_heap(int [], int);
void heap_sort(int [], int);

static void swap(int* a, int* b)
{
    int temp = *b;
    *b = *a;
    *a = temp;
}
 
int parent(int i)
{
    return (int)floor((i - 1) / 2);
}
 
int left(int i)
{
    return (2 * i + 1);
}
 
int right(int i)
{
    return (2 * i + 2);
}
 
/*
 * 堆调整。
 * 以 left(i) 和 right(i) 为根的俩棵子树都已经是堆,
 * 这时A[i]可能小于其子女,所以要进行调整。
 *
 * 调整结果是,以A[i]为根的子树成为一个堆。
 * */
void heapify(int A[], int heap_size, int i)
{
    int l = left(i);
    int r = right(i);
    int largest;
    int temp;
    if (l < heap_size && A[l] > A[i])  {
        largest = l;
    }
    else {
        largest = i;
    }

    if (r < heap_size && A[r] > A[largest])  {
        largest = r;
    }

    if (largest != i) {
        swap(A+i, A+largest);
        heapify(A, heap_size, largest);
    }
}
/* 
 * 将数组A调整为一个堆。
 * 数组A[parent(n-1)]之后的元素都是堆树中的叶子节点。
 * */
void build_heap(int A[], int heap_size)
{
    int i = parent(heap_size-1);
    for (; i >= 0; i--) 
        heapify(A, heap_size, i);
}
/*
 * 每次将堆中最大的元素
 * */
void heap_sort(int A[], int n)
{
    /* 使A[0]称为最大的元素 */
    build_heap(A, n);

    /* 将堆中最后一个叶子节点去掉,还是一个堆,只是堆的大小发生改变。
     * 去掉A[i]后的堆称为新堆,新堆的大小为i。
     * 将新堆的根节点A[0]改为A[i],这时堆的性质发生改变,需要重新调整。
     * 新堆是A[0...i-1], A[i]用来保存原堆的根节点。
     * */
    for(int i = n-1; i >= 0; i--) {

        swap(A, A+i);

        heapify(A, i, 0);
    }
}

void print_heap(int A[], int n)
{
    for (int i = 0; i < n; i++) {
        printf("%d ", A[i]);
    }
    printf("\n");
}

#define NUMOF(a) sizeof(a)/sizeof(a[0])
/*輸入資料並做堆積排序*/
int main(int argc, char* argv[])
{
    int A[] = {19, 1, 10, 14, 16, 4, 7, 9, 3, 2, 8, 5, 11};
    heap_sort(A, NUMOF(A));
    print_heap(A, NUMOF(A));
    return 0;
}

 

高效尤先级队列

主要應用堆的下面兩個操作實現。

 #define MAX_HEAP_NUM 100

#define ERR_HEAP_DATA -1
typedef int HeapDat;
typedef struct heap_tag {
    int size;
    HeapDat data[MAX_HEAP_NUM];
} Heap;

int heap_insert(Heap *heap, HeapDat new_data)
{
    if(heap->size < MAX_HEAP_NUM) {
        HeapDat * heapdata = heap->data;
        int i = heap->size++;

        for (; i>=0 && heapdata[parent(i)] < new_data; i=parent(i))
            heapdata[i] = heapdata[parent(i)];
        heapdata[i] = new_data;

        return 0;
    }
    else 
        return -1;
}


HeapDat heap_extract_max(Heap *heap)
{
    HeapDat max = heap->data[0];
    if (heap->size < 1)
        return ERR_HEAP_DATA;
    heap->size--;
    heap->data[0] = heap->data[heap->size];

    heapify(heap->data, heap->size, 0);
    return max;  
}

int main(int argc, char* argv[]) { Heap h = { 13, {19, 1, 10, 14, 16, 4, 7, 9, 3, 2, 8, 5, 11}, }; build_heap(h.data, h.size); print_heap(h.data, h.size); heap_insert(&h, 18); print_heap(h.data, h.size); heap_extract_max(&h); print_heap(h.data, h.size); return 0; }

 

 另一種建堆的方法:

int _heap_insert(int A[], int n, int new_data, int *heap_size)
{
    if(*heap_size < n) {
        int i = *heap_size;
        for (; i>=0 && A[parent(i)] < new_data; i=parent(i))
            A[i] = A[parent(i)];
        A[i] = new_data;
        *heap_size += 1;
        return 0;
    }
    else 
        return -1;
}


/*
 * 利用heap_insert建堆,跟用上面的方法建的堆不一样。
 * */
void build_heap_2(int A[], int n)
{
    int heap_size=1; 
    do 
        _heap_insert(A, n, A[heap_size], &heap_size);
    while (heap_size < n);
}



/*将A[i] 变为 max(A[i], key); 并重新调整堆
 * */
void heap_increase_key(int A[], int i, int key)
{
    if (key > A[i]) {
        A[i] = key;
        for (; i>=0 && A[parent(i) < key]; i=parent(i)) 
            A[i] = A[parent(i)];
        A[i] = key;
    }
}
/*
 * 将节点A[i]从堆中删除,并重新调整为新的堆。
 * */
int heap_delete(int A[], int heap_size,  int i)
{
    if (i<heap_size) {
        A[i] = A[heap_size-1];
        heapify(A, heap_size, i);
        return 1;
    }
    else
        return 0;
}

 

posted on 2013-02-27 22:44  mhgu  阅读(178)  评论(0编辑  收藏  举报