八、堆

优先队列(Priority Queue): 特殊的“队列”,取出元素的顺序是依照元素的优先权(关键字)大小,而不是元素进入队列的先后顺序。

若采用数组或链表实现优先队列
数组:
插入一元素总是插入尾部 ~Θ(1)
删除一查找最大(或最小)关键字 ~Θ(n)
从数组中删去需要移动元素 ~O(n)
链表:
插入一元素总是插入链表的头部 ~Θ(1)
删除一查找最大(或最小)关键字 ~Θ(n)
删去结点 ~Θ(1)
有序数组:
插入 - 找到合适的位置 ~ O(n)或O(log2N)
移动元素并插入 ~O(n)
删除 - 删去最后一个元素 ~Θ(1)
有序链表:
插入- 找到合适的位置 ~O(n)
插入元素 ~Θ(1)
删除- 删除首元素或最后元素 ~Θ(1)

  • 我们用优先队列的完全二叉树表示

堆的两个特性:

  • 结构性:用数组表示的完全二叉树;
  • 有序性:任意结点的关键字是其子树所有节点的最大值(或最小值)
       “最大堆(MaxHeap)”,也称“大顶堆”:最大值
       “最小堆(MinHeap)”,也称“小顶堆”:最小值

最大堆:


最小堆:

typedef int ElementType;
typedef struct HeapStruct{
    ElementType *elements;
    int size; //堆的当前元素个数
    int capacity; //堆的最大容量
} *MaxHeap;
#define MaxData 10000
//创建最大堆
MaxHeap create(int MaxSize)
{  //创建容量为MaxSize的空间的最大堆
    MaxHeap h = malloc(sizeof(struct HeapStruct));
    h->elements = malloc((MaxSize + 1)*sizeof(ElementType));
    h->size = 0;
    h->capacity = MaxSize;
    //定义“哨兵”为大于堆中所有可能元素的值,以便于以后更快操作
    h->elements[0] = MaxData;
    return h;
}
//判断堆满
bool isFull(MaxHeap h)
{
    return h->size == h->capacity;
}
//判空
bool isEmpty(MaxHeap h)
{
    return h->size == 0;
}

最大堆插入

//往堆里插入元素 T=O(logN)
void insert(MaxHeap h, ElementType item)
{
    int i;
    if(isFull(h))
    {
        printf("最大堆已满");
        return;
    }
    
    i = ++h->size; //i指向插入后堆中最后一个元素的位置
    for (; h->elements[i/2] < item; i/=2) {
        h->elements[i] = h->elements[i/2];
    }
    
    h->elements[i] = item;
}

以上可见插入操作时间复杂度为T = O(logN)

最大堆删除

//删除最大值
ElementType deleteMax(MaxHeap h)
{
    if(isEmpty(h)){
        printf("最大堆已为空");
        return -1;
    }
    ElementType maxItem = h->elements[1];
    ElementType item = h->elements[h->size--];
    int parent = 1, child;
    for (; 2*parent <= h->size; parent = child) {
        child = 2*parent ;
        
        if(child != h->size && h->elements[child + 1] > h->elements[child])
            child++; //child指向左右子结点较大者
        if(h->elements[child] <= item)
            break;
        else{
            h->elements[parent] = h->elements[child];
        }
    }
    h->elements[parent] = item;
    return maxItem;
}

以上可见删除操作时间复杂度为T = O(logN)

最大堆的构建

如果我们用最大堆插入的方式构建最大堆,其时间复杂度为T = O(NlogN)。
我们可以先将数据输入到数组,然后再调整。其时间复杂度为T = O(N/2logN)。

void MaxHeapAdjust(MaxHeap h, int p);
//构建堆
MaxHeap buildMaxHeap()
{
    MaxHeap h = create(10);
    int arr[] = {49,38,65,97,76,13,34,27,49,11};
    for (int i = 1; i <= h->capacity; i++) {
        h->elements[i] = arr[i - 1];
        h->size++;
    }
    /**
     调整堆
     思路跟删除堆相同
     从 h->size/2 的位置往前依次调整堆
     */
    for(int i = h->size/2; i > 0; i--)
    {
        MaxHeapAdjust(h, i);
    }
    return h;
}
//调整堆
void MaxHeapAdjust(MaxHeap h, int p)
{
    int parent = p, child;
    int item = h->elements[p];
    for (; parent * 2 <= h->size; parent = child) {
        child = 2 * parent;
        if(child != h->size && h->elements[child + 1] > h->elements[child])
            child++;
        
        if(item >= h->elements[child])
            break;
        else
            h->elements[parent] = h->elements[child];
    }
    
    h->elements[parent] = item;
}

最大堆堆排序

我们直接调用删除堆,返回最大值的方式,就可以获取从大到小的排序。

//堆排序
void MaxHeapSort(MaxHeap h)
{
    printf("堆排序:");
    for (int i = 1; i <= h->capacity; i++) {
        ElementType item = deleteMax(h);
        printf("%d ",item);
    }
}

调用堆排序

 MaxHeap h = buildMaxHeap();
 MaxHeapSort(h);
执行结果:堆排序:97 76 65 49 49 38 34 27 13 11 
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容