[题解]排队

题意

有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;
    }
};
全部评论

相关推荐

10-25 00:32
香梨想要offer:感觉考研以后好好学 后面能乱杀,目前这简历有点难
点赞 评论 收藏
分享
点赞 收藏 评论
分享
牛客网
牛客企业服务