当前位置: 首页 > 知识库问答 >
问题:

提取最小元素后的最小堆问题

杨乐意
2023-03-14

我正在研究最小堆实现,对这个概念非常陌生。

以此作为参考:< br > https://www.geeksforgeeks.org/building-heap-from-array/
https://algorithm tutor . com/Data-Structures/Tree/Binary-Heaps/

我修改了代码并想出了:

(这是我遇到问题的代码,所有其他代码都与我的问题无关,至少我是这样认为的)

#define LCHILD(x) (2 * (x)) + 1
#define RCHILD(x) (2 * (x)) + 2
#define PARENT(x) ((x) - 1) / 2
typedef struct {
    int key;
    Event *element; // Assume NULL for this example
} Node;
void swap(Node **x, Node **y) {
    Node *temp = *x;
    *x = *y;
    *y = temp;
}
void heapify(void *pq, int n, int i) {
    Node **node = (Node**) pq;

    int smallest = i; // Initialize smallest as root 
    int left = LCHILD(i);
    int right = RCHILD(i); // right = 2*i + 2 
  
    if (left < n && (node[left])->key < (node[smallest ])->key) 
        smallest = left; 
  
    if (right < n && (node[right])->key < (node[smallest ])->key) 
        smallest = right; 
  
    if (smallest != i) { 
        swap(&node[i], &node[smallest ]); 
        heapify(node, n, smallest ); 
    } 
} 

int extractKey(void *pq, int *n) {
    Node **node = (Node**) pq;

    int minElement = (node[0])->key;

    node[0] = node[*n - 1];
    *n = *n - 1;

    heapify(pq, *n, 0);
    return minElement;
}
void insert(void *pq, int key, void *element, int *n) {
    Node **node = (Node**) pq;

    node[*n]->key = key;
    node[*n]->element = element;
    *n = *n + 1;

    int i = *n - 1;
    while ( (i != 0) && (node[PARENT(i)]->key > node[i]->key) ) {
        swap(&node[PARENT(i)], &node[i]);
        i = PARENT(i);
    }
}
  
int main() { 

    Node **pq = malloc (100 * sizeof(Node*));
    int i;
    for(i = 0; i < 100; i++) {
        pq[i] = malloc(sizeof(Node));
        pq[i]->element = malloc(sizeof(Event));
    }
  
    int n = 0; // heap size
  
    insert(pq, 0, NULL, &n);
    printHeap(pq, n); 
    
    insert(pq, 5, NULL, &n);
    printHeap(pq, n); 
    
    insert(pq, 1, NULL, &n);
    printHeap(pq, n); 
    
    insert(pq, 50, NULL, &n);
    printHeap(pq, n); 

    extractKey(pq, &n);
    printHeap(pq, n); 
    
    insert(pq, 10, NULL, &n);
    printHeap(pq, n); 

    return 0;
}

输出:

Array representation of heap is:
0 
Array representation of heap is:
0 5 
Array representation of heap is:
0 5 1 
Array representation of heap is:
0 5 1 50 
Array representation of heap is:
1 5 50 
Array representation of heap is:
1 5 10 10 // What happened here?

仅当我提取最小节点然后添加一个新节点时,才会发生这种情况。如果我不提取节点,那么它工作得很好。我错过了什么吗?

编辑1:这是我正在使用的打印功能。忘了在初始帖子中添加它

(这是此处找到的修改版本:
https://algorithmtutor.com/Data-Structures/Tree/Binary-Heaps/)

void printHeap(void *pq, int n) {
    Node **node = (Node**) pq;

    printf("Array representation of heap is:\n");
  
    for (int i = 0; i < n; ++i) { 
        printf("%d ", node[i]->key);
    }
     printf("\n");
} 

编辑2:我做了更多的测试。这是我得到的:

void insert(void *pq, int key, void *element, int *n) {
    Node **node = (Node**) pq;
    if(*n > 0) {
       printf("node[%d] = %d\n", *n-1, node[*n-1]->key);
    }

    node[*n]->key = key;
    printf("node[%d] = %d\n", *n, node[*n]->key);
    if(*n > 0) {
       printf("node[%d] = %d\n", *n-1, node[*n-1]->key);
    }
    node[*n]->element = element;
    *n = *n + 1;

    // move up until the heap property satisfies
    int i = *n - 1;
    while ( (i != 0) && (node[PARENT(i)]->key > node[i]->key) ) {
        swap(&node[PARENT(i)], &node[i]);
        i = PARENT(i);
    }
}

输出:

node[0] = 0
Array representation of heap is:
0 
node[0] = 0
node[1] = 5
node[0] = 0
Array representation of heap is:
0 5 
node[1] = 5
node[2] = 1
node[1] = 5
Array representation of heap is:
0 5 1 
node[2] = 1
node[3] = 50
node[2] = 1
Array representation of heap is:
0 5 1 50 
Array representation of heap is:
1 5 50 
node[2] = 50
node[3] = 10
node[2] = 10 // Huh? it should be 50
Array representation of heap is:
1 5 10 10 

共有1个答案

宋伟泽
2023-03-14

问题是节点[0]=节点[*n-1];行。这是将两个节点指针设置为相同的值,因此您不再有100个唯一的节点指针。(因此,它也泄漏了内存。)将行更改为交换(

 类似资料:
  • 本摘录自《破解编码访谈》第5版 数字随机生成并存储在一个(扩展的)数组中。你如何跟踪中位数 作者向我们介绍了一个基于堆的解决方案: 堆非常擅长基本排序和跟踪max和mins。这实际上很有趣——如果你能跟踪元素的大一半和小一半。大一半保存在最小堆中,这样大一半中的最小元素在根。小一半保存在最大堆中,这样最小一半中的最大元素在根。现在有了这些数据结构,你就有了潜在的中值元素在根。”如果堆不再相同大小,

  • 我真的很困惑Kth最小元素和Kth元素之间到底有什么区别。 第k个元素=第k个元素是数组=数组[k-1] 但是,什么是第k个最小元素?我有一个家庭作业问题,我需要写一个算法来找到2个排序数组中的第k个最小元素。我不是来要求你做作业的,你不需要给我任何算法或代码。我只想理解它是什么意思。Kth最小元素和kth元素之间有什么不同。 我问这个问题的原因是:我在谷歌上搜索什么是第k个最小的元素,其中一个是

  • 我实现了c程序,可以找到矩阵的元素:行的最大元素,同时列的最小元素,或行的-min元素,同时列的最大元素。例如,我们有数据。包含以下内容的txt文件: 4 7 8 9 10 6 5 4 11 5 0 1 12 4 2 7 13- 其中4是n-矩阵大小(4x4),7和10是这些数字。 下面是代码: 问题:我想知道我的代码是不是“脏”代码?因为我总是渴望让一切变得如此困难,只要有可能让它变得容易。是否

  • 我试图找到最小元素并删除它,但不幸的是我不能。我想得到一些帮助,这是我的代码。 我在Class Stack中只有这些方法:equals,is空,pop,推送,top,toString(主要方法)。 提前感谢。

  • 本文向大家介绍JavaScript 查找最小或最大元素,包括了JavaScript 查找最小或最大元素的使用技巧和注意事项,需要的朋友参考一下 示例 如果您的数组或类似数组的对象是numeric,也就是说,如果它的所有元素都是数字,则可以使用Math.min.apply或作为第一个参数Math.max.apply传递null,而将数组作为第二个参数传递。 6 在ES6中,可以使用...运算符扩展数

  • 问题内容: 您是否知道一个流行的库(Apache,Google等),该库具有可靠的最小- 最大堆Java实现,即允许在其中查看其最小值和最大值并删除其中的元素的堆? 问题答案: 番石榴:。