补充一个题意不太清楚的地方:蔬菜坏掉是假设蔬菜都有标号,那么特定标号的蔬菜就会在特定时间坏掉。如果你及时卖了它们,那么那一天就不会有新的蔬菜坏掉。

结论1:如果我们知道了k天的答案,那么我们直接扔掉若干个最小的蔬菜即可获得k - 1天的答案。

证:因为能在k天卖的一定能在k - 1天卖.....

推论1:只要求100000天卖哪些蔬菜即可。

结论2:最贵的菜越晚卖越好。

证:因为它最贵,所以是一定要卖的。我们要让别的菜尽量卖的多,晚卖有决策包容性。

推论2:按照从贵到廉考虑蔬菜,每个菜越晚卖越好。

结论3:可以把每个蔬菜按照变质天数分类,当天变质的蔬菜能够出售的时间是第一天到当天。

推论3:假如把每种菜按照变质天数分类,那么有额外奖励的蔬菜应该在最后一天。

证:因为额外奖励比正常蔬菜贵,所以一定会优先卖它。由结论2可知应放在最后。

然后算法就出来了。分类之后按顺序放蔬菜。每天维护从它开始向前,第一个空闲的地方。可以用并查集实现。

每天能放多少菜要特别注意,代码实现上我用总数 - 之前应该留下的菜数来限制。

 #include <bits/stdc++.h>

 inline char gc() {
/*static char buf[1000000], *p1, *p2;
if(p1 == p2) p2 = (p1 = buf) + fread(buf, 1, 1000000, stdin);
return (p1 == p2) ? EOF : *p1++;*/ return getchar();
} template <class T> inline void read(T &x) {
x = ;
char c = gc();
bool f = ;
while(c < '' || c > '') {
if(c == '-') f = ;
c = gc();
}
while(c >= '' && c <= '') {
x = x * + c - ;
c = gc();
}
if(f) x = (~x) + ;
return;
} typedef long long LL;
const int N = ; struct Node {
int val, d, id, cnt, flag;
Node(int V = , int D = , int ID = , int C = , int F = ) {
val = V;
d = D;
id = ID;
cnt = C;
flag = F;
}
inline bool operator <(const Node &w) const {
if(val != w.val)
return val < w.val;
return d < w.d;
}
}; std::priority_queue<Node> Q, Q2;
int a[N], s[N], c[N], x[N], fa[N], rest[N];
LL ans[N]; int find(int x) {
if(x == fa[x]) return x;
return fa[x] = find(fa[x]);
} inline void del(LL &a) {
Node t = Q2.top();
Q2.pop();
a += t.val;
t.cnt--;
if(t.cnt) Q2.push(t);
return;
} int main() { int n, m, k, lm = ;
read(n); read(m); read(k);
for(int i = ; i <= lm; i++) {
fa[i] = i;
rest[i] = m;
}
for(int i = ; i <= n; i++) {
read(a[i]); read(s[i]); read(c[i]); read(x[i]);
LL day;
if(!x[i]) {
day = lm;
}
else {
day = std::min((c[i] - ) / x[i] + , lm);
}
Q.push(Node(a[i] + s[i], day, i, , ));
if((day - ) * x[i] + < c[i]) { /// the last day have rest
Q.push(Node(a[i], day, i, c[i] - - x[i] * (day - ), ));
}
if(day > && x[i]) {
Q.push(Node(a[i], day - , i, x[i] * (day - ), ));
}
//c[i]--;
} //int Last = 0, Cnt = 0; int tot = ;
while(Q.size()) {
Node t = Q.top();
Q.pop();
int now = find(t.d);
while(now && t.cnt) {
LL large;
if(t.flag) large = std::min(t.cnt, std::min(t.cnt - x[t.id] * (now - ), rest[now]));
else if(x[t.id]) large = std::min(t.cnt, std::min(x[t.id], rest[now]));
else large = std::min(t.cnt, rest[now]); ans[lm] += t.val * large;
tot += large;
Q2.push(Node(-t.val, , t.id, large));
rest[now] -= large;
t.cnt -= large;
if(!rest[now]) {
fa[now] = find(now - );
}
now = find(now - );
}
} int day = (tot - ) / m + ; /// need (day) to sell the vegetables for(int i = lm - ; i >= day; i--) {
ans[i] = ans[lm];
}
ans[day - ] = ans[day];
for(int T = ; T <= tot - m * (day - ); T++) {
del(ans[day - ]);
} for(int i = day - ; i >= ; i--) {
ans[i] = ans[i + ];
for(int T = ; T <= m; T++) {
del(ans[i]);
}
} for(int i = ; i <= k; i++) {
int t;
read(t);
printf("%lld\n", ans[t]);
} return ;
}

AC代码

LOJ#2306 蔬菜的更多相关文章

  1. *LOJ#2306. 「NOI2017」蔬菜

    $n \leq 100000$种蔬菜,每个蔬菜有:一单位价格:卖第一单位时额外价格:总量:每天腐烂量.每天能卖$m \leq 10$单位蔬菜,多次询问:前$k \leq 100000$天最多收入多少. ...

  2. 【LOJ】#2306. 「NOI2017」蔬菜

    题解 从后往前递推 如果我们知道了第i天的最优方案和第i天选择的蔬菜,加入第i天选择的蔬菜数量为S,我们只需要减去最小的S - (i - 1) * M 个蔬菜即可 所以我们只要求出最后一天的蔬菜选择 ...

  3. bzoj4946 Noi2017 蔬菜

    题目描述 小 N 是蔬菜仓库的管理员,负责设计蔬菜的销售方案. 在蔬菜仓库中,共存放有nn 种蔬菜,小NN 需要根据不同蔬菜的特性,综合考虑各方面因素,设计合理的销售方案,以获得最多的收益. 在计算销 ...

  4. [Noi2016]区间 BZOJ4653 洛谷P1712 Loj#2086

    额... 首先,看到这道题,第一想法就是二分答案+线段树... 兴高采烈的认为我一定能AC,之后发现n是500000... nlog^2=80%,亲测可过... 由于答案是求满足题意的最大长度-最小长 ...

  5. Loj #2192. 「SHOI2014」概率充电器

    Loj #2192. 「SHOI2014」概率充电器 题目描述 著名的电子产品品牌 SHOI 刚刚发布了引领世界潮流的下一代电子产品--概率充电器: 「采用全新纳米级加工技术,实现元件与导线能否通电完 ...

  6. Loj #3096. 「SNOI2019」数论

    Loj #3096. 「SNOI2019」数论 题目描述 给出正整数 \(P, Q, T\),大小为 \(n\) 的整数集 \(A\) 和大小为 \(m\) 的整数集 \(B\),请你求出: \[ \ ...

  7. Loj #3093. 「BJOI2019」光线

    Loj #3093. 「BJOI2019」光线 题目描述 当一束光打到一层玻璃上时,有一定比例的光会穿过这层玻璃,一定比例的光会被反射回去,剩下的光被玻璃吸收. 设对于任意 \(x\),有 \(x\t ...

  8. Loj #3089. 「BJOI2019」奥术神杖

    Loj #3089. 「BJOI2019」奥术神杖 题目描述 Bezorath 大陆抵抗地灾军团入侵的战争进入了僵持的阶段,世世代代生活在 Bezorath 这片大陆的精灵们开始寻找远古时代诸神遗留的 ...

  9. Loj #2542. 「PKUWC2018」随机游走

    Loj #2542. 「PKUWC2018」随机游走 题目描述 给定一棵 \(n\) 个结点的树,你从点 \(x\) 出发,每次等概率随机选择一条与所在点相邻的边走过去. 有 \(Q\) 次询问,每次 ...

随机推荐

  1. day 7-3 僵尸进程,孤儿进程与守护进程

    一.基本定义 正常情况下,子进程是通过父进程创建的,子进程在创建新的进程.子进程的结束和父进程的运行是一个异步过程,即父进程永远无法预测子进程 到底什么时候结束. 当一个 进程完成它的工作终止之后,它 ...

  2. Programming好文解读系列(—)——代码整洁之道

    注:初入职场,作为一个程序员,要融入项目组的编程风格,渐渐地觉得系统地研究下如何写出整洁而高效的代码还是很有必要的.与在学校时写代码的情况不同,实现某个功能是不难的,需要下功夫的地方在于如何做一些防御 ...

  3. (C/C++)区别:数组与指针,指针与引用

    1.数组跟指针的区别 数组要么在静态存储区被创建(如全局数组),要么在栈上被创建.数组名对应着(而不是指向)一块内存,其地址与容量在生命期内保持不变,只有数组的内容可以改变. 指针可以随时指向任意类型 ...

  4. python爬虫之Splash使用初体验

    Splash是什么: Splash是一个Javascript渲染服务.它是一个实现了HTTP API的轻量级浏览器,Splash是用Python实现的,同时使用Twisted和QT.Twisted(Q ...

  5. windows 10 & task view & shortcut

    windows 10 & task view & shortcut Win + Tab https://blogs.windows.com/windowsexperience/2014 ...

  6. Google浏览器解决编码乱码问题

    新版google浏览器编码乱码没有设置的入口,怎么办呢?. 步骤一: 可以下载goole的插件,名为charset,下载后的文件名为Charset_v0.4.1 步骤二: google右上角-> ...

  7. java 中 Math类

    package cn.liuliu.com; import java.math.BigDecimal; import java.math.BigInteger; /* * 一.Math类? * * 1 ...

  8. 10.Service资源发现

    Kubernetes Pods是不可控的.每当一个pod停止后,他不是重启,而是重建.ReplicaSets特别是Pods动态地创建和销毁(例如,当向外扩展或向内扩展时).虽然每个PodIP地址都有自 ...

  9. jsp页面给字体加颜色

    jsp页面给字体加颜色<span style="color:red">要加颜色的部分</span>

  10. gym-101350M

    题意:给你一堆货币汇率,再给你一堆货币,算下值多少钱: 思路:直接map搞定: #include<iostream> #include<algorithm> #include& ...