码迷,mamicode.com
首页 > 其他好文 > 详细

[CF521D]Shop

时间:2018-09-27 22:17:30      阅读:267      评论:0      收藏:0      [点我收藏+]

标签:har   fun   \n   源代码   air   可以转化   int   main   sum   

[CF521D]Shop

题目大意:

你有一个长度为\(k(k\le10^5)\)的数列\(A_{1\sim k}\),有\(n(n\le10^5)\)种操作,操作包含以下\(3\)种:

  1. \(A_x\)变成\(y\)
  2. \(A_x\)加上\(y\)
  3. \(A_x\)乘以\(y\)

定义这个数列的收益为各项之积,你可以从中选\(m\)个操作(每个操作至多选\(1\)次),使得收益最大,求任一操作的方案。

思路:

贪心,对于只有操作\(3\)的情况,显然从大到小贪心更优。

而操作\(2\)可以转化成操作\(3\),操作\(1\)又可以转化成操作\(2\),因此最后还是可以用同样的方式贪心。

源代码:

#include<queue>
#include<cstdio>
#include<cctype>
#include<vector>
#include<algorithm>
#include<functional>
inline int getint() {
    register char ch;
    while(!isdigit(ch=getchar()));
    register int x=ch^'0';
    while(isdigit(ch=getchar())) x=(((x<<2)+x)<<1)+(ch^'0');
    return x;
}
typedef long long int64;
const int N=1e5+1;
int ans[N];
std::pair<int,int> b[N];
struct Modify {
    int a,b,c;
    bool operator > (const Modify &rhs) const {
        return c>rhs.c;
    }
};
Modify q[N];
std::vector<std::pair<int,int> > v[N];
struct Node {
    int id;
    double c;
    bool operator < (const Node &rhs) const {
        return c<rhs.c;
    }
};
std::priority_queue<Node> h;
inline bool cmp(const int &i,const int &j) {
    return q[i].a<q[j].a;
}
int main() {
    const int k=getint(),n=getint(),m=getint();
    for(register int i=1;i<=k;i++) {
        v[i].push_back(std::make_pair(getint(),0));
    }
    for(register int i=1;i<=n;i++) {
        q[i].a=getint();
        q[i].b=getint();
        q[i].c=getint();
        if(q[i].a==1) {
            b[q[i].b]=std::max(b[q[i].b],std::make_pair(q[i].c-v[q[i].b][0].first,i));
        }
        if(q[i].a==3) {
            h.push((Node){i,1.*q[i].c});
        }
    }
    for(register int i=1;i<=k;i++) {
        if(b[i].first>0) v[i].push_back(b[i]);
    }
    for(register int i=1;i<=n;i++) {
        if(q[i].a==2) {
            v[q[i].b].push_back(std::make_pair(q[i].c,i));
        }
    }
    for(register int i=1;i<=k;i++) {
        std::sort(v[i].begin()+1,v[i].end(),std::greater<std::pair<int,int> >());
        int64 sum=v[i][0].first;
        for(register unsigned j=1;j<v[i].size();j++) {
            h.push((Node){v[i][j].second,1.*(sum+v[i][j].first)/sum});
            sum+=v[i][j].first;
        }
    }
    for(register int i=1;!h.empty()&&i<=m;i++) {
        ans[++ans[0]]=h.top().id;
        h.pop();
    }
    if(ans[0]==0) {
        puts("0");
        return 0;
    }
    std::sort(&ans[1],&ans[ans[0]]+1,cmp);
    for(register int i=0;i<=ans[0];i++) {
        printf("%d%c",ans[i]," \n"[!(i%ans[0])]);
    }
    return 0;
}

[CF521D]Shop

标签:har   fun   \n   源代码   air   可以转化   int   main   sum   

原文地址:https://www.cnblogs.com/skylee03/p/9715477.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!