使用C++构建一个优先级队列的实现

 更新时间:2025年02月06日 09:04:16   作者:zhoudeng666  
优先级队列是一种特殊的队列数据结构,本文主要介绍了使用C++构建一个优先级队列的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1.优先级队列的介绍

优先级队列是一种特殊的队列数据结构,它是队列,但又不完全是,因为它要将装载的数据进行优先级排序,找到一个最大或者最小优先级的元素,下一次出队列的元素就是这个元素,所以说它不完全是一个队列,优先级队列广泛应用于任务调度、资源分配、事件处理、Dijkstra算法、A*搜索算法等领域。

2.优先级队列的设计

 个人感觉,设计一个优先级队列就是一个堆建立和调整的过程,因为要装载元素,所以我们采用vector来作为成员变量,后续就是对与这个vector变量的管理就行了,因为我们只是简单设计一个优先级队列,所以并没有设计在优先级队列中传入函数进行比较的过程,只是简单的比较大小的过程,首先我们要设计向上调整函数和向下调整函数,因为这个的构造函数和析构函数非常简单,这里我就不过多赘述了。

向上调整函数

向上调整函数就是子节点与父节点做比较,这里我们假设建小堆,如果子节点小于父节点,那么这个堆就是有问题的,所以要进行交换,将子节点与父节点进行交换,但是我们又要考虑交换以后得子节点也就是现在的父节点是否和他现在的父节点又是不对的关系的,所以我们要进行循环,只有当节点为0,也就是根节点或者,子节点小于父节点时,进行循环退出

void siftUp(size_t index)
{
    
    while (index > 0)
    {
        int father = index / 2;
        
        if (list[father] > list[index])
        {
            swap(list[father], list[index]);
            index = father;
        }
        else
        {
            break;
        }
    }
}

向下调整函数

向下调整就是判断你是否比你的两个孩子都要小,逻辑和向上调整函数差不多,只是要和两个孩子都比一次,或者找最小的那个孩子比,主要是要知道孩子是index/2-1和index/2-2就行

void siftDown(size_t index)
{
    while (index <= list.size())
    {
        leftson = index * 2 + 1;
        rightson = index * 2 + 2;
        if (list[index] > list[leftson])
        {
            swap(list[index], list[leftson])
                index = leftson;
        }
        else if (list[index] > list[rightson])
        {
            swap(list[index], list[rightson]);
            index = rightson;
        }
        else
        {
            break;
        }
    }
}

建堆函数

在构造函数的第一步肯定是拷贝构造对应的vector,然后得到的这个vector大概率不是一个堆,因此要对它进行调整,使得它可以成为一个符合规则的堆,一般建堆有两种策略,一种是自下而上的sift down方法,另一种是自上而下的sift up方法。这里我们使用sit down 向下调整的方法,找到最后一个非叶子节点,也就是list.size()/2,然后对每个非叶子节点进行向下调整

explicit Heap(const std::vector<T>& array)
    :list(array)
{
    buildHeap();
}
void buildHeap()
{
    for (int i = list.size() / 2 ; i >=0  ; i--)
    {
        siftDown(i);
   }
}

插入函数

当我们插入一个元素时,我们首先是尾插入vector列表,然后对这个元素进行向上调整,就使得这个堆变得规则

void insert(const T& value)
{
    list.insert(value);
    siftUp(list.size() - 1);
}

获取顶端元素函数

我们建立这个堆或者说维护这个优先级队列肯定是为了得到一个有价值的元素,所有这个堆里面最有价值的元素当然时堆顶元素,但是当我们得到堆顶元素时,这个堆也会被破坏,所以,我们一般会将堆顶元素和最后一个元素进行交换,然后再对堆顶元素进行向下调整,就让堆保持合适的规则

void removeTop()
{
    if (list.empty()) {
        throw std::out_of_range("Heap is empty");
    }
    swap(list[0], list[list.size() - 1]);
    siftDown(0);
    list.erase(list.end() - 1);
}

到此这篇关于使用C++构建一个优先级队列的实现的文章就介绍到这了,更多相关C++ 优先级队列内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • c++ 数字类型和字符串类型互转详解

    c++ 数字类型和字符串类型互转详解

    今天小编就为大家分享一篇讲解c++ 数字类型和字符串类型互转的文章,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-09-09
  • 详细了解C语言二叉树的建立与遍历

    详细了解C语言二叉树的建立与遍历

    这篇文章主要介绍了C中二叉树的建立和各种遍历实例代码,涉及树节点的定义,后序遍历,层序遍历,深度优先和广度优先等相关内容,具有一定借鉴价值,需要的朋友可以参考下
    2021-07-07
  • 基于linux下获取时间函数的详解

    基于linux下获取时间函数的详解

    本篇文章是对linux下获取时间的函数进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • C++俄罗斯方块游戏 无需图形库的俄罗斯方块

    C++俄罗斯方块游戏 无需图形库的俄罗斯方块

    这篇文章主要为大家详细介绍了无需图形库的C++俄罗斯方块游戏,重温经典游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-06-06
  • 详细分析C++ 多态和虚函数

    详细分析C++ 多态和虚函数

    这篇文章主要介绍了C++ 多态和虚函数的相关资料,文中示例代码非常详细,帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-07-07
  • VS2019创建c++动态链接库dll与调用方法实践

    VS2019创建c++动态链接库dll与调用方法实践

    动态链接库是一个包含可由多个程序同时使用的代码和数据的库,本文主要介绍了VS2019创建c++动态链接库dll与调用方法,具有一定的参考价值,感兴趣的可以了解一下
    2024-06-06
  • 基于C语言实现三子棋游戏

    基于C语言实现三子棋游戏

    这篇文章主要为大家详细介绍了基于C语言实现三子棋游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • 详解C++中函数模板的定义与使用

    详解C++中函数模板的定义与使用

    函数模板实质就是参数化数据类型,称这种编程模式为数据类型泛化编程。本文将通过示例来和大家一起了解下C++中函数模板的定义与使用,需要的可以参考一下
    2022-09-09
  • 深入探索C++中stack和queue的底层实现

    深入探索C++中stack和queue的底层实现

    这篇文章主要介绍了C++中的stack和dequeue的底层实现,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-09-09
  • 浅谈C语言中的指针和数组有什么区别

    浅谈C语言中的指针和数组有什么区别

    C语言中的指针和数组是两个重要的数据结构,它们在内存管理和数据存储方面有许多相似之处,但也存在一些关键的区别,本文就来介绍一下C语言中的指针和数组有什么区别,具有一定的参考价值,感兴趣的可以了解一下
    2023-09-09

最新评论