[USACO06FEB] Stall Reservations 贪心

比较巧的\(O(n)\)扫一遍做法,用一个小跟堆维护所有牛棚最后一个牛占用的时间(即\(r_i\)),贪心的想,如果最先结束的那个牛棚都不能满足当前牛,那么我们只能新开一个牛棚,并继续维护小根堆;如果那个牛棚满足,那么这个牛就去那个牛棚,更新当前牛棚最后占用时间。

#include <cstdio>
#include <queue>
#include <algorithm>
using namespace std;
inline int read(){
    char ch=getchar();int s=0;
    while(ch<'0'||ch>'9') ch=getchar();
    while(ch>='0'&&ch<='9') s=s*10+(ch^'0'), ch=getchar();
    return s;
}
struct nod{
    int cnt,last;
    const bool operator < (const nod &a) const{
        return last>a.last; // 小根堆
    }
    nod(int cnt, int last):cnt(cnt),last(last){}
};
#define MAXN 50005
struct nod2{
    int l,r,id;
    const bool operator < (const nod2 &a) const{
        if(l==a.l) return r<a.r;
        return l<a.l;
    }
} a[MAXN];
int ans[MAXN];
priority_queue<nod> q;
int n;
int sum=0;
int main(){
    n=read();
    for(int i=1;i<=n;++i) a[i].id=i,a[i].l=read(),a[i].r=read();
    sort(a+1, a+1+n);
    q.push(nod(1, a[1].r));
    ans[a[1].id]=1;
    ++sum;
    for(int i=2;i<=n;++i){
        nod cur=q.top();
        if(cur.last>=a[i].l){
            ++sum; // 新开牛棚
            ans[a[i].id]=sum;
            q.push(nod(sum, a[i].r));
        }else{
            q.pop(); // 更新牛棚最后结束时间
            ans[a[i].id]=cur.cnt;
            q.push(nod(cur.cnt, a[i].r));
        }
    }
    printf("%d\n", sum);
    for(int i=1;i<=n;++i) printf("%d\n", ans[i]);
    return 0;
}
01-01 16:46