优先级队列-C语言

参考库函数sort逻辑写的优先级队列,测试基本类型和结构体可以实现功能

代码

c 复制代码
#include <stdio.h>
#include <stdint.h>
#include <string.h>
#include <stdlib.h>
typedef struct Heaps {
    void *heap;
    size_t elemSize;
    int len;
    int (*cmp)(const void *, const void *);
}Heap;

static int ParentIndex(int idex)
{
    return ((idex + 1) >> 1) - 1;
    // return (idex+1)/2-1;
}
static int LeftChildIndex(int idex)
{
    return ((idex + 1) << 1) - 1;
    // return (idex+1)*2-1;
}
static int RightChildIndex(int idex)
{
    return LeftChildIndex(idex) + 1;
}
static void Swap(void *a, void *b, size_t elemSize)
{
    //int tmp = *(int *)a;
    //*(int *)a = *(int *)b;
    //*(int *)b = tmp;
    void *p = malloc(elemSize);
    memcpy(p, a, elemSize);
    memcpy(a, b, elemSize);
    memcpy(b, p, elemSize);
    free(p);
    return;
}
void *FindPointer(Heap *h, int index) {
    return (void *)((const uint8_t *)(h->heap)+((index)*(h->elemSize)));
}
static void Sink(Heap *h)
{
    int pos = 0;
    while (pos < h->len) {
        int leftPos = LeftChildIndex(pos);
        int rightPos = RightChildIndex(pos);
        if (leftPos >= h->len) break;
        int maxPos = pos;
        if((h->cmp)(FindPointer(h, leftPos), FindPointer(h, pos)) < 0) {
            maxPos = leftPos;
        }
        if(rightPos < h->len && ((h->cmp)(FindPointer(h, rightPos), FindPointer(h, maxPos)) < 0)) {
            maxPos = rightPos;
        } 
        if(maxPos != pos) {
            Swap(FindPointer(h, maxPos), FindPointer(h, pos), h->elemSize);
            pos = maxPos;
        }
        else {
            break;
        }
    }
}

static void FloatingUp(Heap *h) 
{
    int pos = h->len-1;
    while(pos > 0) {
        int parentPos = ParentIndex(pos);
        void *p = FindPointer(h, parentPos);
        void *q = FindPointer(h, pos);
        if(((h->cmp)(p, q)) > 0) {
            Swap(p, q, h->elemSize);
            pos = parentPos;
        } 
        else {
            break;
        }
    }
    return;
}

Heap *InitHeap(int k, size_t elemSize, int (*cmp)(const void *, const void *)) {
    Heap *h = (Heap *)malloc(sizeof(Heap));
    h->heap = malloc(elemSize*k);
    h->len = 0;
    h->cmp = cmp;
    h->elemSize = elemSize;
    return h;
}
void HeapPush(Heap *h, void *v) {
    //((int *)h->heap)[h->len++] = v;
    void *p = FindPointer(h, h->len);
    memcpy(p, v, h->elemSize);
    h->len++;
    FloatingUp(h);
}
void *HeapPop(Heap *h) {
    void *res = malloc(h->elemSize);
    memcpy(res, h->heap, h->elemSize);
    void *p = FindPointer(h, h->len-1);
    memcpy(h->heap, p, h->elemSize);
    //((int *)h->heap)[0] = ((int *)h->heap)[h->len-1];
    h->len--;
    Sink(h);
    return res;
}
void HeapDestroy(Heap *h)
{
    free(h->heap);
    free(h);
}

使用

c 复制代码
typedef struct {
	int a;
	int b;
} Elemtype;
// 小顶堆
int cmp(const void *a, const void *b) {
    return ((Elemtype *)a)->a - ((Elemtype *)b)->a;
} 

Elemtype tmp[] = {
	{.a = 3, .b = 5},
	{.a = 6, .b = 7},
	{.a = 1, .b = 8},
};
int n = sizeof(tmp)/sizeof(Elemtype);
// 初始化堆大小,堆元素大小, 比较函数
Heap *h = InitHeap(n, sizeof(Elemtype), cmp);
for(int i=0; i<n; i++) {
	HeapPush(h, (void *)&(tmp[i]));
}
while(h->len > 0) {
	Elemtype *p = (Elemtype *)HeapPop(h);
	printf("%d %d\n", p->a, p->b);
}
HeapDestroy(h);
相关推荐
Yzzz-F1 分钟前
算法竞赛进阶指南 动态规划 背包
算法·动态规划
程序员-King.4 分钟前
day124—二分查找—最小化数组中的最大值(LeetCode-2439)
算法·leetcode·二分查找
阿华hhh7 分钟前
Linux系统编程(网络udp)
linux·服务器·c语言·网络·网络协议·udp
predawnlove9 分钟前
【NCCL】4 AllGather-PAT算法
算法·gpu·nccl
驱动探索者10 分钟前
[缩略语大全]之[内存管理]篇
java·网络·算法·内存管理
风筝在晴天搁浅32 分钟前
hot100 234.回文链表
数据结构·链表
superman超哥33 分钟前
仓颉类型别名的使用方法深度解析
c语言·开发语言·c++·python·仓颉
·云扬·37 分钟前
MySQL Join关联查询:从算法原理到实战优化
数据库·mysql·算法
bbq粉刷匠41 分钟前
二叉树中两个指定节点的最近公共祖先
java·算法
是Yu欸1 小时前
从Ascend C算子开发视角看CANN的“软硬协同”
c语言·开发语言·云原生·昇腾·ascend·cann·开放社区