每日一题 6月15日 Supermarket 优先队列OR并查集维护i前第一个空格

题目链接:https://ac.nowcoder.com/acm/problem/50995
题目大意:
图片说明
思路:贪心+优先队列:我们按时间从小到大排序,当前时间T。并且维护已经选取的商品的最小利润。如果当前商品的时间==T。那么从优先队列来里面找到最小值如果<当前商品的利润:替换。
如果当前商品的时间<T,直接选取。

思路:并查集:我们按商品的利润从大到小排序,那么我们希望从前往后贪心。并且把每个商品放在自己过期时间前越后越好。我们用并查集来维护时间。f[i]=-1:第i天空闲。否则f[x]=y。x前面第一个空闲的天数是f[y]。
如果找到空闲天x.维护f[x]=f[x-1]

1:
#include <bits/stdc++.h>
#define LL long long
using namespace std;

struct node{
    int w, t;
}a[10005];

priority_queue<LL, vector<LL>, greater<LL> > q;
int main(){
    int n;
    while(~scanf("%d", &n)){
        for(int i=1; i<=n; i++){
            scanf("%d%d", &a[i].w, &a[i].t);
        }
        sort(a+1, a+1+n, [](node &a, node &b){return a.t<b.t;});
        LL T=0, ans=0;
        for(int i=1; i<=n; i++){
            if(T<a[i].t){
                q.push(a[i].w);
                T++;
                ans+=a[i].w;
            }
            else if(a[i].w>q.top()){
                ans+=(a[i].w-q.top());
                q.pop(); q.push(a[i].w);
            }
        }
        printf("%lld\n", ans);
        while(!q.empty()){
            q.pop();
        }
    }

    return 0;
}
*****************************
2:
#include <bits/stdc++.h>
#define LL long long
using namespace std;

struct node{
    int w, t;
}a[10005];

int f[10005];
int fd(int x){
    if(f[x]==-1) return x;
    return f[x]=fd(f[x]);
}

int main(){
    int n;
    while(~scanf("%d", &n)){
        memset(f, -1, sizeof(f));
        for(int i=1; i<=n; i++){
            scanf("%d%d", &a[i].w, &a[i].t);
        }
        sort(a+1, a+1+n, [](node &a, node &b){return a.w>b.w;});
        LL ans=0;
        for(int i=1; i<=n; i++){
            int pos=fd(a[i].t);
            if(pos>0){
                ans+=a[i].w;
                f[pos]=pos-1;
            }
        }
        printf("%lld\n", ans);
    }

    return 0;
}
全部评论

相关推荐

牛客120493863号:你姐东南大学硕士在读,那就找导师或者师兄师姐打听下同门同方向前辈就业最好的是去向哪几家公司了呗(如果不想走考公选调的话),这个是最有参考性的。
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务