「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. VS code 同步设置与插件

    准备工作:拥有一个github账户,电脑上需安装VSCode.实现同步的功能主要依赖于VSCode插件 "Settings Sync"第一步:安装同步插件Settings Sync ...

  2. python学习笔记:unittest单元测试

    单元测试:开发自测时写的代码 unittest基本原理: ♦整个平台的搭建使用的是python的unittest测试框架,这里简单介绍下unittest模块的简单应用. ♦unittest是pytho ...

  3. Emqtt集群搭建

    1 Emqtt简单搭建 1.1  介绍:EMQ:EMQ 2.0,号称百万级开源MQTT消息服务器,基于Erlang/OTP语言平台开发,支持大规模连接和分布式集群,发布订阅模式的开源MQTT消息服务器 ...

  4. sqlalchemy的不区分大小写比较

    方法一:collation 参照:https://segmentfault.com/q/1010000010203547 方法是在 db.String 中添加 collation='NOCASE' 描 ...

  5. JS formData

    有时候,我们需要使用formData来上传文件,并根据反馈的结果做不同的处理, 这个时候使用form来提交文件不方便 #perform 获取的的上传文件所用的input标签 var formData ...

  6. Linux下查看日志文件

    问题一:日志文件打开,卡死 在公司开发的时候,开发的程序在本地测试正常,但是在测试环境却有问题.这个时候第一反应就是查看日志文件,看看日志文件里面有什么错误信息.我潇洒的执行了一下 vim log.t ...

  7. Java常见问题收集

    转载处:https://blog.csdn.net/jackfrued/article/details/44921941 1.面向对象的特征有哪些方面? 答:面向对象的特征主要有以下几个方面: - 抽 ...

  8. BFS+打印路径

    题目是给你起点sx,和终点gx:牛在起点可以进行下面两个操作: 步行:John花一分钟由任意点X移动到点X-1或点X+1. 瞬移:John花一分钟由任意点X移动到点2*X. 你要输出最短步数及打印路径 ...

  9. 使用form提交到搜狗浏览器示例

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  10. NGUI的输入框制作(attach- input filed script的使用)

    一,我们添加一个sprite,给这个sprite添加一个box collider ,然后添加input filed script,如下图: 二,我们给sprite添加一个child的label,然后绑 ...