如何获取优先级队列的最后一个元素而不遍历它

时间:2015-03-30 18:38:20

标签: c++ queue priority-queue

我有以下优先级队列:

#include <iostream>
#include <queue>
#include <iomanip>

using namespace std;

struct Time {
    int h; // >= 0
    int m; // 0-59
    int s; // 0-59
};

class CompareTime {
public:
    bool operator()(Time& t1, Time& t2)
    {
       if (t1.h < t2.h) return true;
       if (t1.h == t2.h && t1.m < t2.m) return true;
       if (t1.h == t2.h && t1.m == t2.m && t1.s < t2.s) return true;
       return false;
    }
};

int main()
{
    priority_queue<Time, vector<Time>, CompareTime> pq;

    // Array of 4 time objects:

    Time t[4] = { {3, 2, 40}, {3, 2, 26}, {5, 16, 13}, {5, 14, 20}};

    for (int i = 0; i < 4; ++i)
       pq.push(t[i]);
    while (! pq.empty()) {
       Time t2 = pq.top();
       cout << setw(3) << t2.h << " " << setw(3) << t2.m << " " <<
       setw(3) << t2.s << endl;
       pq.pop();
    }

    return 0;
}

现在为了获取最后一个元素,我必须pop()队列中的所有元素。是否有某种方法可以检索此优先级队列的最后一个元素。

我知道可以在&#34; CompareTime&#34;中反转顺序。以便最后一个元素成为第一个元素。我不想这样做,因为我想按照&#34; CompareTime&#34;确定的顺序pop()优先级队列中的元素。但与此同时,我还想要优先级队列的最后一个元素。不要弹出优先级队列中的所有元素。是否可以确定优先级队列的最后一个元素的值。

我正在使用以下gcc编译器:gcc(Ubuntu / Linaro 4.6.4-6ubuntu2)4.6.4

3 个答案:

答案 0 :(得分:3)

std :: priority_queue不支持您想要直接执行的操作。请参阅: http://www.cplusplus.com/reference/queue/priority_queue/

当然,您可以随时创建自己的DS,但有一些库容器可以满足您的需求。

编辑:

您可以考虑使用

std::set 

http://en.cppreference.com/w/cpp/container/set

订购时,您可以快速访问第一个和最后一个元素,也可以快速删除任何元素。请注意,元素必须是唯一的。如果你不想要,你可以使用

std::multiset 

更灵活,可以做到这一点。

获取两端的元素

std::set::begin
std::set::rbegin

迭代器可用。由于集合始终排序,因此这些是最小元素和最大元素。

希望有所帮助!

答案 1 :(得分:1)

您可以将值存储为某个较大的值 - 实际值 然后在检索时执行相同的操作,即大值 - 队列(顶部)值。 例如 比如说,大值= 100 然后存储 56为44 20为80 ... 并在检索时 80为100-80 = 20

这是你可以按反向优先顺序填充队列的方法;)

答案 2 :(得分:0)

构造两个优先级队列,而不是一个,像往常一样使用第一个,然后将所有true更改为false,反之亦然,因此,第一个优先级队列的最后一个将成为第二个优先级队列的第一个。