POJ 3481 Double Queue(set实现)

Double Queue

  The new founded Balkan Investment Group Bank (BIG-Bank) opened a new office in Bucharest, equipped with a modern computing environment provided by IBM Romania, and using modern information technologies. As usual, each client of the bank is identified by a positive integer Kand, upon arriving to the bank for some services, he or she receives a positive integer priority P. One of the inventions of the young managers of the bank shocked the software engineer of the serving system. They proposed to break the tradition by sometimes calling the serving desk with the lowest priority instead of that with the highest priority. Thus, the system will receive the following types of request:

0The system needs to stop serving
KPAdd client K to the waiting list with priority P
2Serve the client with the highest priority and drop him or her from the waiting list
3Serve the client with the lowest priority and drop him or her from the waiting list

  Your task is to help the software engineer of the bank by writing a program to implement the requested serving policy.

Input

  Each line of the input contains one of the possible requests; only the last line contains the stop-request (code 0). You may assume that when there is a request to include a new client in the list (code 1), there is no other request in the list of the same client or with the same priority. An identifier K is always less than 106, and a priority P is less than 107. The client may arrive for being served multiple times, and each time may obtain a different priority.

Output

  For each request with code 2 or 3, the program has to print, in a separate line of the standard output, the identifier of the served client. If the request arrives when the waiting list is empty, then the program prints zero (0) to the output.

Sample Input

2
1 20 14
1 30 3
2
1 10 99
3
2
2
0

Sample Output

0
20
30
10
0

解题思路:
  本题要求处理银行排队问题,根据每行的输入处理数据;若输入为1则继续输入两个整数,以第一个数为权值,第二个数为优先级(数值越大优先级越高)加入排队系统中;若输入为2 则处理当前优先级最高的用户,并输出用户的权值;若输入为3 则处理当前优先级最低的用户,并输出用户权值;若输入为0结束运行。若当没有排队人数则无论输入2、3都输出0。

  利用set自带红黑树排序的性质,可以在集合中直接获取最高与最低优先级,用pair记录权值和优先级,将pair传入集合。用迭代器可以获得集合首位(最大优先级)和集合末尾(最小优先级),数据处理完毕后可以通过erase()删除当前用户以便后续操作。

样例分析:
Sample Input

2          //处理高优先级用户,当前无人排队输出0
1 20 14      //优先级为14的用户20开始排队 
1 30 3       //优先级为3的用户30开始排队
2          //处理高优先级用户,当前队列中有 20 14   30 3 最高优先级14 输出20 
1 10 99      //优先级为99的用户10开始排队 
3          //处理低优先级用户,当前队列中有 10 99   30 3 最低优先级3 输出30
2          //处理高优先级用户,当前队列中有 10 99 最高优先级99 输出10
2          //处理高优先级用户, 当前无人排队输出0
0          //输入为0结束

 1 #include <cstdio>
 2 #include <iterator>
 3 #include <set>
 4 //bits/stdc++.h编译错误
 5 using namespace std;
 6 //利用set的排序性质
 7 struct cmp{ //set比较方法以pair的第二个成员为基准,数值越大的在前
 8     bool operator() (const pair<int, int> &a, const pair<int, int> &b)
 9     {
10         return a.second > b.second;
11     }
12 };
13 set<pair<int, int>, cmp> waitingList;  //用set记录当前队列
14 int main()
15 {
16     int n;
17     while(scanf("%d", &n) != EOF){  //输入数据
18         if(n == 0)  //输入为0结束运行
19             break;
20         if(n == 1){ //输入为1
21             pair<int, int> temp;
22             scanf("%d%d", &temp.first, &temp.second);   //记录用户权值与优先级
23             waitingList.insert(temp);   //用户开始排队
24         }
25         if(n == 2){ //输入为2
26             if(!waitingList.empty()){
27                 set<pair<int, int>, cmp>::iterator it = waitingList.begin();
28                 //获取优先级最高的用户
29                 printf("%d\n", (*it).first);    //输出用户权值
30                 waitingList.erase(it);  //删除用户
31             }else{
32                 printf("0\n");  //队中无人
33             }
34         }
35         if(n == 3){
36             if(!waitingList.empty()){
37                 set<pair<int, int>, cmp>::iterator it = waitingList.end();
38                 it--;
39                 //获取优先级最小的用户
40                 printf("%d\n", (*it).first);    //输出用户权值
41                 waitingList.erase(it);  //删除用户
42             }else{
43                 printf("0\n");//队中无人
44             }
45         }
46     }
47     return 0;
48 }

 

转载于:https://www.cnblogs.com/suvvm/p/9886565.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是 poj3714 突袭的代码实现,使用了 Kruskal 算法求解最小生成树: ```c++ #include <iostream> #include <algorithm> #include <vector> using namespace std; const int MAXN = 20005; const int MAXM = 100005; struct Edge { int u, v, w; bool operator<(const Edge& other) const { return w < other.w; } }; int n, m, p; int parent[MAXN], depth[MAXN]; Edge edges[MAXM]; void make_set(int v) { parent[v] = v; depth[v] = 0; } int find_set(int v) { if (v == parent[v]) { return v; } return parent[v] = find_set(parent[v]); } void union_sets(int a, int b) { a = find_set(a); b = find_set(b); if (a != b) { if (depth[a] < depth[b]) { swap(a, b); } parent[b] = a; if (depth[a] == depth[b]) { depth[a]++; } } } int kruskal() { int ans = 0; for (int i = 1; i <= n; i++) { make_set(i); } sort(edges, edges + m); for (int i = 0; i < m; i++) { int u = edges[i].u, v = edges[i].v, w = edges[i].w; if (find_set(u) != find_set(v)) { union_sets(u, v); ans += w; } } return ans; } int main() { while (cin >> n >> m >> p) { for (int i = 0; i < m; i++) { int u, v, w; cin >> u >> v >> w; edges[i] = {u, v, w}; } int ans1 = kruskal(); for (int i = 1; i <= n; i++) { make_set(i); } for (int i = 0; i < p; i++) { int u, v, w; cin >> u >> v >> w; edges[i] = {u, v, -w}; } int ans2 = kruskal(); cout << ans2 - ans1 << endl; } return 0; } ``` 在这个实现中,使用了一个 `Edge` 结构体来表示一条边,包括起点、终点和边权。然后使用 Kruskal 算法求解最小生成树,分别计算突袭前和突袭后的最小生成树的权值和,最终答案为突袭后的权值和减去突袭前的权值和。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值