[题解]排队
题意
有n个任务,m个处理任务的队列。任务处理时间为a[1],...,a[n]。按顺序从1-n处理任务。问有多少对满足第i个任务完成时间严格大于第j个任务完成的时间。
题解
关键词
优先队列
树状数组
做法一
分为两步,第一步模拟求出结束的顺序,即得到一个数组b,b[i]表示第i号是第b[i]个办理完成的。这部分可模拟完成。
然后遍历,得到b[i]>b[j]的数量。这部分也是。
总复杂度,预计通过30%。
做法二
考虑加速做法一的两个部分。
模拟部分可以使用优先队列进行加速。的复杂度得到数组b。
求解,b[i]>b[j]的数量可以通过树状数组(或归并排序求逆序对的方法)加速。复杂度也是。
总复杂度,预计通过100%。
代码
#include <bits/stdc++.h> #define ui unsigned int #define ll long long #define ull unsigned ll using namespace std; const int maxn = 100100; struct node { int id; ll t; bool operator < (const node & a) const { return a.t < t || (a.t == t && a.id < id); } }; int f[maxn],ed[maxn]; class Solution { public: /** * 求解合法的(i,j)对的数量 * @param n int整型 n个人 * @param m int整型 m个窗口 * @param a int整型vector 长度为n的vector,顺序表示1-n号客人的办理业务所需时间 * @return long长整型 */ int lowbit(int x) {return x&-x;} void upd(int i, int k, int n) { for (;i<=n;i+=lowbit(i)) f[i]+=k; } int ask(int i, int n) { int res=0; for (;i>0;i-=lowbit(i)) res += f[i]; return res; } long long getNumValidPairs(int n, int m, vector<int>& a) { priority_queue<node>q; int left = m, idx = 0; ll st = 0, ans = 0; for (int i=1;i<=n;i++) { if (left == 0) { node x = q.top(); q.pop(); ed[x.id] = ++idx; upd(x.id,-1,n); ans += ask(x.id,n); st = x.t; } else left--; q.push((node){i, st+a[i-1]}); upd(i,1,n); } while (!q.empty()) { node x = q.top(); q.pop(); ed[x.id] = ++idx; upd(x.id,-1,n); ans += ask(x.id,n); } return ans; } };