LOJ#2306 蔬菜
补充一个题意不太清楚的地方:蔬菜坏掉是假设蔬菜都有标号,那么特定标号的蔬菜就会在特定时间坏掉。如果你及时卖了它们,那么那一天就不会有新的蔬菜坏掉。
结论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 蔬菜的更多相关文章
- *LOJ#2306. 「NOI2017」蔬菜
$n \leq 100000$种蔬菜,每个蔬菜有:一单位价格:卖第一单位时额外价格:总量:每天腐烂量.每天能卖$m \leq 10$单位蔬菜,多次询问:前$k \leq 100000$天最多收入多少. ...
- 【LOJ】#2306. 「NOI2017」蔬菜
题解 从后往前递推 如果我们知道了第i天的最优方案和第i天选择的蔬菜,加入第i天选择的蔬菜数量为S,我们只需要减去最小的S - (i - 1) * M 个蔬菜即可 所以我们只要求出最后一天的蔬菜选择 ...
- bzoj4946 Noi2017 蔬菜
题目描述 小 N 是蔬菜仓库的管理员,负责设计蔬菜的销售方案. 在蔬菜仓库中,共存放有nn 种蔬菜,小NN 需要根据不同蔬菜的特性,综合考虑各方面因素,设计合理的销售方案,以获得最多的收益. 在计算销 ...
- [Noi2016]区间 BZOJ4653 洛谷P1712 Loj#2086
额... 首先,看到这道题,第一想法就是二分答案+线段树... 兴高采烈的认为我一定能AC,之后发现n是500000... nlog^2=80%,亲测可过... 由于答案是求满足题意的最大长度-最小长 ...
- Loj #2192. 「SHOI2014」概率充电器
Loj #2192. 「SHOI2014」概率充电器 题目描述 著名的电子产品品牌 SHOI 刚刚发布了引领世界潮流的下一代电子产品--概率充电器: 「采用全新纳米级加工技术,实现元件与导线能否通电完 ...
- Loj #3096. 「SNOI2019」数论
Loj #3096. 「SNOI2019」数论 题目描述 给出正整数 \(P, Q, T\),大小为 \(n\) 的整数集 \(A\) 和大小为 \(m\) 的整数集 \(B\),请你求出: \[ \ ...
- Loj #3093. 「BJOI2019」光线
Loj #3093. 「BJOI2019」光线 题目描述 当一束光打到一层玻璃上时,有一定比例的光会穿过这层玻璃,一定比例的光会被反射回去,剩下的光被玻璃吸收. 设对于任意 \(x\),有 \(x\t ...
- Loj #3089. 「BJOI2019」奥术神杖
Loj #3089. 「BJOI2019」奥术神杖 题目描述 Bezorath 大陆抵抗地灾军团入侵的战争进入了僵持的阶段,世世代代生活在 Bezorath 这片大陆的精灵们开始寻找远古时代诸神遗留的 ...
- Loj #2542. 「PKUWC2018」随机游走
Loj #2542. 「PKUWC2018」随机游走 题目描述 给定一棵 \(n\) 个结点的树,你从点 \(x\) 出发,每次等概率随机选择一条与所在点相邻的边走过去. 有 \(Q\) 次询问,每次 ...
随机推荐
- Oracle 修改数据库表数据提交之后进行回滚
--查看历史数据 select * from test1 as of timestamp to_timestamp('2018-12-23 14:41:00', 'yyyy-mm-dd hh24:mi ...
- 在 Ubuntu14.04 上搭建 Spark 2.3.1(latest version)
搭建最新的 Spark 2.3.1 . 首先需要下载最新版 jdk .目前 2.3.1 需要 8.0 及其以上 jdk 才可以允许. 所以如果你没有 8.0 jdk 安装好了之后会报错.不要尝试安装 ...
- python数据结构与算法第十三天【归并排序】
1.代码实现 def merge_sort(alist): if len(alist) <= 1: return alist # 二分分解 num = len(alist)/2 left = m ...
- fastjson与各类型的转换
参考:https://www.cnblogs.com/ceshi2016/p/7381478.html http://www.cnblogs.com/goody9807/p/4244862.html ...
- 去掉AMD锐龙和Intel Kaby Lake的不支持的硬件的提示
Windows 7和Windows 8.1都不支持AMD Ryzen 锐龙系列和Intel最新的Kaby Lake系列,Windows Update 更新之后总是会提示“不支持的硬件(Unsuppor ...
- [oracle] to_date() 与 to_char() 日期和字符串转换
to_date("要转换的字符串","转换的格式") 两个参数的格式必须匹配,否则会报错. 即按照第二个参数的格式解释第一个参数. to_char(日期,& ...
- 11/5/2018模拟 Problem C
题面 题解 我有特殊的哈希技巧 以到下一个相同字符的距离为值哈希, 如果不存在或在串外, 就是 \(|T| + 1\). 加入一个新字符 \(S_i\) 时, 同时修改它上一次出现时的值, 由 \(| ...
- Jquery实现检测用户输入用户名和密码不能为空
要求 1.用户名和密码为空点击登录时提示相应的提示 2.获取用户名输入框时,错误提示清除 思路 1.创建1个input-text标签和1个input-password标签,1个input-botton ...
- 了解AutoCAD对象层次结构 —— 1 ——应用程序
想象这样一个场景:当您开始一天的工作,坐到电脑前面,用鼠标双击桌面上的AutoCAD Civil 3D图标,这时一个AutoCAD Civil 3D应用程序将运行起来.打开Windows任务管理器,我 ...
- redis哨兵集群+spring boot 2.×
Ubuntu集群构建篇 redis-cli:不跟参数,默认访问localhost:6379端口,无密码登陆 redis-cli -h ${host} -p ${port} -a ${password} ...