「NOI2017」蔬菜

首先考虑流

可以从 \(s\) 流入表示得到蔬菜,流出到 \(t\) 表示卖出蔬菜,给每个蔬菜拆点,并给它它每天应得的蔬菜。

但是我们没办法直接给,注意到如果把变质看成得到并可以留给上一天,我们每天就可以得到变质的蔬菜并获得从后一天没用完的蔬菜,这就是建图的大体思路。

然后你发现这个东西需要对询问天数动态加点,加点后发现需要退流,可以暴力退 \(m\) 的流,复杂度是正确的。

期望得分 \(60\) 分

然后研究一下,发现退流是没有必要的,也就是说第 \(i\) 天选择的蔬菜一定是 第 \(i+1\) 天的子集

然后你可以写的简单一点。


考虑实际上,费用流的过程是可以模拟的,或者直接从贪心出发。

最后一天的选择集合是确定的,并且选择集合只会扩大,不会缩小,然后问题其实就变成了模拟。

你只需要拿一个堆维护当前可选的蔬菜的集合(注意,蔬菜的大小在外面维护即可)

对于一血,可以拆成新的一个蔬菜,也可以在进堆的时候特判

然后可以对每个询问暴力模拟,得到复杂度 \(O(nmq\log n)\) 的做法

期望得分 \(80\) 分


然后注意到第 \(i\) 天可以从第 \(i+1\) 天进行递推

维护一个第 \(i+1\) 天的买的集合,如果集合大小大于 \(mi\) ,就删掉小的删到 \(mi\)

复杂度 \(O(nm\log n)\) ,期望得分 \(100\) 分


Code:

#include <cstdio>
#include <cctype>
#include <cstring>
#include <queue>
#include <vector>
#include <algorithm>
#define ll long long
using std::min;
using std::max;
const int SIZE=1<<21;
char ibuf[SIZE],*iS,*iT;
//#define gc() (iS==iT?(iT=(iS=ibuf)+fread(ibuf,1,SIZE,stdin),iS==iT?EOF:*iS++):*iS++)
#define gc() getchar()
template <class T>
void read(T &x)
{
int f=0;x=0;char c=gc();
while(!isdigit(c)) f|=c=='-',c=gc();
while(isdigit(c)) x=x*10+c-'0',c=gc();
if(f) x=-x;
}
const int N=2e5+10;
int n,m,k,a[N],s[N],c[N],x[N];
std::vector<int> seg[N];
struct node
{
int id,c;
node(){}
node(int a,int b){id=a,c=b;}
bool friend operator <(node a,node b){return a.c<b.c;}
};
std::priority_queue <node> q;
int sta[N],tot,yuu[N*10],sell[N],vis[N],mxt;
ll ans[N];
int main()
{
//freopen("vegetables2.in","r",stdin);
//freopen("vegetables2.out","w",stdout);
read(n),read(m),read(k);
for(int i=1;i<=n;i++)
{
read(a[i]),read(s[i]),read(c[i]),read(x[i]);
//单位收益,一血收益,总库存,每天变质
if(x[i]) mxt=max(mxt,(c[i]-1)/x[i]+1);
}
mxt=min(mxt,100000);
if(!mxt) mxt=100000;
mxt+=5000;
for(int i=1;i<=n;i++)
{
if(x[i]) seg[min(mxt,(c[i]-1)/x[i]+1)].push_back(i);
else seg[mxt].push_back(i);
}
for(int sel,i=mxt;i;i--)
{
for(int j=0;j<seg[i].size();j++)
{
int id=seg[i][j];
q.push(node(id,a[id]+s[id]));
}
int lim=m;
while(lim&&!q.empty())
{
int now=q.top().id;
q.pop();
if(!vis[now])
{
--lim;
ans[mxt]+=a[now]+s[now];
++sell[now];
if(sell[now]!=c[now]) q.push(node(now,a[now]));
yuu[++yuu[0]]=a[now]+s[now];
vis[now]=1;
}
else
{
sel=min(lim,c[now]-x[now]*(i-1)-sell[now]);
sell[now]+=sel;
lim-=sel;
if(sell[now]!=c[now]) sta[++tot]=now;
ans[mxt]+=1ll*a[now]*sel;
for(int j=1;j<=sel;j++) yuu[++yuu[0]]=a[now];
}
}
for(int j=1;j<=tot;j++) q.push(node(sta[j],a[sta[j]]));
tot=0;
}
std::sort(yuu+1,yuu+1+yuu[0]);
for(int j=1,i=mxt;i;i--)
{
ans[i-1]=ans[i];
while(yuu[0]-j+1>(i-1)*m) ans[i-1]=ans[i-1]-yuu[j++];
}
for(int p,i=1;i<=k;i++) read(p),printf("%lld\n",ans[min(p,mxt)]);
return 0;
}

2019.6.24

「NOI2017」蔬菜 解题报告的更多相关文章

  1. 「NOI2017」游戏 解题报告

    「NOI2017」游戏 \(d\)这么小,你考虑直接对\(d\)个东西暴力 枚举\(x\)为\(a\)或\(b\)(\(c\)就不用了,因为\(a,b\)已经包含\(c\))了,剩下的就是个\(2-s ...

  2. 「NOI2017」整数 解题报告

    「NOI2017」整数 有一些比较简单的\(\log^2n\)做法 比如暴力在动态开点线段树上维护每个位置为\(0\)还是\(1\),我们发现涉及到某一位加上\(1\)或者减去\(1\)实际上对其他位 ...

  3. 「ZJOI2016」旅行者 解题报告

    「ZJOI2016」旅行者 对网格图进行分治. 每次从中间选一列,然后枚举每个这一列的格子作为起点跑最短路,进入子矩形时把询问划分一下,有点类似整体二分 至于复杂度么,我不会阿 Code: #incl ...

  4. 「HNOI2016」树 解题报告

    「HNOI2016」树 事毒瘤题... 我一开始以为每次把大树的子树再接给大树,然后死活不知道咋做,心想怕不是个神仙题哦 然后看题解后才发现是把模板树的子树给大树,虽然思维上难度没啥了,但是还是很难写 ...

  5. 「HNOI2016」序列 解题报告

    「HNOI2016」序列 有一些高妙的做法,懒得看 考虑莫队,考虑莫队咋移动区间 然后你在区间内部找一个最小值的位置,假设现在从右边加 最小值左边区间显然可以\(O(1)\),最小值右边的区间是断掉的 ...

  6. 「HNOI2016」网络 解题报告

    「HNOI2016」网络 我有一个绝妙的可持久化树套树思路,可惜的是,它的空间是\(n\log^2 n\)的... 注意到对一个询问,我们可以二分答案 然后统计经过这个点大于当前答案的路径条数,如果这 ...

  7. 「HAOI2018」染色 解题报告

    「HAOI2018」染色 是个套路题.. 考虑容斥 则恰好为\(k\)个颜色恰好为\(c\)次的贡献为 \[ \binom{m}{k}\sum_{i\ge k}(-1)^{i-k}\binom{m-k ...

  8. 「HNOI2016」最小公倍数 解题报告

    「HNOI2016」最小公倍数 考虑暴力,对每个询问,处理出\(\le a,\le b\)的与询问点在一起的联通块,然后判断是否是一个联通块,且联通块\(a,b\)最大值是否满足要求. 然后很显然需要 ...

  9. 「SCOI2016」围棋 解题报告

    「SCOI2016」围棋 打CF后困不拉基的,搞了一上午... 考虑直接状压棋子,然后发现会t 考虑我们需要上一行的状态本质上是某个位置为末尾是否可以匹配第一行的串 于是状态可以\(2^m\)压住了, ...

随机推荐

  1. 牛客提高D6t1 积木大赛

    分析 每次修改用二位差分记录一下 之后对于三维分别统计即可 代码 #include<iostream> #include<cstdio> #include<cstring ...

  2. HDU 1269 迷宫城堡 (Kosaraju)

    题目链接:HDU 1269 Problem Description 为了训练小希的方向感,Gardon建立了一座大城堡,里面有N个房间(N<=10000)和M条通道(M<=100000), ...

  3. Angular2+之模态框-使用ngx-bootstrap包中的模态框组件实现

    模态框是项目中经常会用到的一个公共功能,通常会被用左提示框或者扩展选项框. 下面,我用一个小例子来简单展示实现模态框功能的过程: 1.为项目加包: ng add ngx-bootstrap 2.在xx ...

  4. RMQ(连续相同最大值)

    http://poj.org/problem?id=3368 Frequent values Time Limit: 2000MS   Memory Limit: 65536K Total Submi ...

  5. python 模块和包深度学习理解

    python 模块和包 简单说相当于命名空间 1,python 模块        python模块就是一个文件,里面有函数,变量等 import 模块 模块.方法 from 模块 import fu ...

  6. Kafka、ActiveMQ、RabbitMQ、RocketMQ 都有什么区别,消息队列有什么优点和缺点

    面试题 为什么使用消息队列? 消息队列有什么优点和缺点? Kafka.ActiveMQ.RabbitMQ.RocketMQ 都有什么区别,以及适合哪些场景? 面试官心理分析 其实面试官主要是想看看: ...

  7. NGUI的anchors属性的使用

    一,anchors锚点 我们需要明白target目标的使用,这时是你下面使用left,right,bottom和top的距离,比如我们使用目标为UI Root,这个就是摄像机的视野,所以,我们使用an ...

  8. MySQL对字段新增自增序列

    现在有这样的场景,我们的数据库类型是MySQL,表是从其他库拿过来的,约束和索引都没迁移.现在希望增加一个自增序列. 且自增序列是从当前最大自增ID开始的,下面就是这样一个过程的演示. mysql&g ...

  9. Python-编码这趟浑水

    最近听Alex讲到python编码,还特意用博客讲解,觉得问题严重了,于是翻看各种博客,先简单的对编码错误做一个总结,其他的后续慢慢补上,还得上班.还得学习.还得写博客?感觉有点吃不消了.各位大神不喜 ...

  10. 免费资源(CDN,顶级域名)汇集

    CloudFlare:免费CDN,需要将域名指向到cloudflare服务器.付费的可以使用二级域名 https://www.cloudflare.com/ Freenom:freenom会提供免费提 ...