Description

http://www.lydsy.com/JudgeOnline/upload/Noi2017D2.pdf

Solution

网上大部分都是并查集写法,但是有大神写了非并查集写法,特别容易理解

首先 \(s_i\) 的限制,只需将每一个蔬菜分出一个价值为 \(a_i+s_i\) 且过期时间为该蔬菜最晚的一天的蔬菜

把时间倒序之后,问题转化为每个蔬菜会在第几天出现,每天贪心选择价值最大的即可

先求出 \(\max \{p_i \}\) 的答案,然后递推 \([1,\max \{pi\}-1]\) 的答案:每次删除价值最小的 \(m\) 单位蔬菜,不难发现所有蔬菜的销售时间依然合法

#include<bits/stdc++.h>
#define ui unsigned int
#define ll long long
#define db double
#define ld long double
#define ull unsigned long long
const int MAXN=100000+10,P=100000;
int n,m,k,a[MAXN],c[MAXN],x[MAXN],s[MAXN],vis[MAXN];
ll ans[MAXN],sale[MAXN],tot;
struct node{
ll id,val;
inline bool operator < (const node &A) const {
return val<A.val;
};
inline bool operator > (const node &A) const {
return val>A.val;
};
};
std::vector<int> V[MAXN];
std::queue<node> red;
std::priority_queue<node> q1;
std::priority_queue< node,std::vector<node>,std::greater<node> > q2;
template<typename T> inline void read(T &x)
{
T data=0,w=1;
char ch=0;
while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
if(ch=='-')w=-1,ch=getchar();
while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
x=data*w;
}
template<typename T> inline void write(T x,char ch='\0')
{
if(x<0)putchar('-'),x=-x;
if(x>9)write(x/10);
putchar(x%10+'0');
if(ch!='\0')putchar(ch);
}
template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
template<typename T> inline T min(T x,T y){return x<y?x:y;}
template<typename T> inline T max(T x,T y){return x>y?x:y;}
int main()
{
read(n);read(m);read(k);
for(register int i=1;i<=n;++i)
{
read(a[i]),read(s[i]),read(c[i]),read(x[i]);
if(x[i]==0)V[P].push_back(i);
else V[min(P,(c[i]+x[i]-1)/x[i])].push_back(i);
}
for(register int i=P;i>=1;--i)
{
for(register int j=0,lt=V[i].size();j<lt;++j)q1.push({V[i][j],a[V[i][j]]+s[V[i][j]]});
ll rest=m;
while(!q1.empty()&&rest)
{
node now=q1.top();
q1.pop();
if(vis[now.id])
{
ll res=min(rest,c[now.id]-(i-1)*x[now.id]-sale[now.id]);
ans[P]+=1ll*res*a[now.id];sale[now.id]+=res;rest-=res;
if(sale[now.id]<c[now.id])red.push(now);
}
else
{
vis[now.id]=1;
ans[P]+=now.val;sale[now.id]++;rest--;
if(sale[now.id]<c[now.id])q1.push((node){now.id,a[now.id]});
}
}
while(!red.empty())q1.push(red.front()),red.pop();
}
for(register int i=1;i<=n;tot+=sale[i],++i)
if(sale[i]==1)q2.push((node){i,a[i]+s[i]});
else if(sale[i])q2.push((node){i,a[i]});
for(register int i=P-1;i>=1;--i)
{
ans[i]=ans[i+1];
ll rest=tot-1ll*m*i;
if(rest<=0)continue;
else
while(!q2.empty()&&rest)
{
node now=q2.top();
q2.pop();
if(sale[now.id]==1)sale[now.id]--,rest--,ans[i]-=now.val;
else
{
int res=min(rest,sale[now.id]-1);
sale[now.id]-=res,rest-=res,ans[i]-=1ll*res*now.val;
if(sale[now.id]==1)q2.push((node){now.id,a[now.id]+s[now.id]});
else if(sale[now.id])q2.push((node){now.id,a[now.id]});
}
}
tot=1ll*m*i;
}
for(register int i=1,x;i<=k;++i)read(x),write(ans[x],'\n');
return 0;
}

【刷题】BZOJ 4946 [Noi2017]蔬菜的更多相关文章

  1. BZOJ.4946.[NOI2017]蔬菜(贪心 离线)

    题目链接 因为有删除,考虑倒序处理某个p的询问. 那么每天删除xi的蔬菜就变成了每天运来xi的蔬菜.那么我们取当前最优的即可,早取晚取都一样,不需要留给后面取,还能给后面更优的留出空间. 这样就只需考 ...

  2. 4946: [Noi2017]蔬菜

    4946: [Noi2017]蔬菜 http://www.lydsy.com/JudgeOnline/upload/Noi2017D2.pdf 分析: 贪心. 首先可以将一个蔬菜拆成两个,一个是有加成 ...

  3. [NOI2017]蔬菜 贪心

    题面: [NOI2017]蔬菜 题解: 首先每天蔬菜会变质这点并不好处理,我们考虑让时间倒流,从后向前处理,这样的话就相当于每天都会得到一定量的蔬菜. 这样做有什么好处呢? 我们可以发现一个性质:如果 ...

  4. 【刷题】BZOJ 2407 探险

    Description 探险家小T好高兴!X国要举办一次溶洞探险比赛,获奖者将得到丰厚奖品哦!小T虽然对奖品不感兴趣,但是这个大振名声的机会当然不能错过! 比赛即将开始,工作人员说明了这次比赛的规则: ...

  5. 【刷题】BZOJ 4543 [POI2014]Hotel加强版

    Description 同OJ3522 数据范围:n<=100000 Solution dp的设计见[刷题]BZOJ 3522 [Poi2014]Hotel 然后发现dp的第二维与深度有关,于是 ...

  6. 【刷题】BZOJ 4316 小C的独立集

    Description 图论小王子小C经常虐菜,特别是在图论方面,经常把小D虐得很惨很惨. 这不,小C让小D去求一个无向图的最大独立集,通俗地讲就是:在无向图中选出若干个点,这些点互相没有边连接,并使 ...

  7. 【刷题】BZOJ 4176 Lucas的数论

    Description 去年的Lucas非常喜欢数论题,但是一年以后的Lucas却不那么喜欢了. 在整理以前的试题时,发现了这样一道题目"求Sigma(f(i)),其中1<=i< ...

  8. BZOJ第一页刷题计划

    BZOJ第一页刷题计划 已完成:67 / 90 [BZOJ1000]A+B Problem:A+B: [BZOJ1001][BeiJing2006]狼抓兔子:最小割: [BZOJ1002][FJOI2 ...

  9. 【刷题】BZOJ 2260 商店购物

    Description Grant是一个个体户老板,他经营的小店因为其丰富的优惠方案深受附近居民的青睐,生意红火.小店的优惠方案十分简单有趣.Grant规定:在一次消费过程中,如果您在本店购买了精制油 ...

随机推荐

  1. flask前端与后端之间传递的两种数据格式:json与FormData

    json格式 双向! 前端 ==>后端:json格式 后端 ==>前端:json格式 html <!-- html部分 --> <form enctype='applic ...

  2. how2j 的shiro教程初探

    教程案例里的mysql连接器只支持mysql,不支持mariadb,如果用的不是mysql,创建连接时会报错.

  3. Java虚拟机笔记(五):JVM中对象的分代

    为什么要分代 为什么需要把堆分代?不分代不能完成他所做的事情么?其实不分代完全可以,分代的唯一理由就是优化GC性能.你先想想,如果没有分代,那我们所有的对象都在一块,GC的时候我们要找到哪些对象没用, ...

  4. PS官方正式中文版(搬砖分享)

    https://pan.baidu.com/s/1c3IdQq0 PS官方正式中文版(搬砖分享) 注意事项: 1.安装开始前请先断网,在成功破解激活前请全程断网: 2.安装完成后先试运行软件一次,然后 ...

  5. mark一下岗位

    一.中国移动杭州研发中心——测试开发工程师 https://campusresume.zhaopin.com/resume/14375/1   等内推信息 岗位描述:作为产品的质量守护者,在全面理解被 ...

  6. 加速github、kaggle访问、加速python packge下载更改源

    OS: WIN10 加速github.kaggle访问 使用站长DNS工具(http://tool.chinaz.com/dns) 查询响应速度最快的网站服务器IP,将网站服务器IP和域名添加到电脑h ...

  7. Bitcoin挖矿

    目录 为什么要挖矿? 比特币挖矿 为什么要挖矿? 增加恶意行为的成本 增加记账权力,获取相应的奖励 比特币挖矿 每开采210000个区块,挖矿奖励减半 2009年1月-2012年11月,奖励50BTC ...

  8. Vigenere加密

    Vigenere加密法原理很简单,实现起来也不难.与普通的单码加密法不同,明文经过加密之后,每个字母出现的频率就不会有高峰和低峰. 密钥中字母代表行和明文中的字母代表行.在vigenere表中找到对应 ...

  9. 微软职位内部推荐-Service Engineer for Office365

    微软近期Open的职位: Key Responsibilities: The Service Engineer in this team will be responsible for plannin ...

  10. ejs 模板使用方法

    http://embeddedjs.com/ Embedded JS Templates Embedded JS(EJS) 来源于ERB模板,且与ERB有很多相似之处.它有着与ERB相同的Tag,且包 ...