BZOJ4380 Myjnie / Luogu3592 [POI2015]MYJ-区间DP
Description
有$n$家洗车店从左往右排成一排,每家店都有一个正整数价格$p[i]$。
有$m$个人要来消费,第$i$个人会驶过第$a[i]$个开始一直到第$b[i]$个洗车店,且会选择这些店中最便宜的一个进行一次消费。但是如果这个最便宜的价格大于$c[i]$,那么这个人就不洗车了。
请给每家店指定一个价格,使得所有人花的钱的总和最大。
Solution
神仙$DP$ QAQ
每个店的价格肯定是$c_i$中的某一个值, 所以可以离散化
定义状态 $dp[L][R][k]$ 表示 在区间$[i,j]$ 最小值为$k$ 时所能收益的最大值
转移 : $dp[L][R][k] = \max{(dp[L][i - 1][k] + dp[i + 1][R][k] + cnt[i][k])}$
但是发现这样无法快速求出答案, 所以把$dp[L][R][k]$定义为 最小值 $>=k$时所能收益的最大值。
则多了一个转移: $dp[L][R][k] = \max{(dp[L][R][k], dp[L][R][k + 1])}$。
题目要求 求出方案, 则定义$val[L][R][k]$ 为真正的$k$值, $P[L][R][k]$ 为哪个位置取 $k$
最后递归求方案
空间复杂度$O(N^2M)$, 时间复杂度$O(N^3M)$
Code
#include<cstdio>
#include<cstring>
#include<algorithm>
#define N 55
#define M 4005
#define rd read()
using namespace std; int n, m;
int dp[N][N][M], cnt[N][M], val[N][N][M], P[N][N][M];
int ls[M], tot, ans[N]; struct node {
int l, r, val;
}a[M]; inline int read() {
int X = , p = ; char c = getchar();
for (; c > '' || c < ''; c = getchar())
if (c == '-') p = -;
for (; c >= '' && c <= ''; c = getchar())
X = X * + c - '';
return X * p;
} int fd(int x) {
return lower_bound(ls + , ls + + tot, x) - ls;
} void cmax(int &A, int B) {
if (A < B)
A = B;
} void DP(int L, int R) {
memset(cnt, , sizeof(cnt));
for (int i = ; i <= m; ++i) {
if (a[i].l < L || a[i].r > R)
continue;
for (int j = a[i].l; j <= a[i].r; ++j)
cnt[j][a[i].val]++;
}
for (int i = L; i <= R; ++i)
for (int j = tot - ; j; --j)
cnt[i][j] += cnt[i][j + ];
for (int i = tot; i; --i) {
int maxn = -, pos = ;
for (int j = L; j <= R; ++j) {
int res = dp[L][j - ][i] + dp[j + ][R][i] + cnt[j][i] * ls[i];
if (res > maxn)
maxn = res, pos = j;
cmax(dp[L][R][i], res);
}
val[L][R][i] = i;
P[L][R][i] = pos;
if(i < m && dp[L][R][i] < dp[L][R][i + ])
val[L][R][i] = val[L][R][i + ],
dp[L][R][i] = dp[L][R][i + ],
P[L][R][i] = P[L][R][i + ];
}
} void findans(int L, int R, int lim) {
if (L > R) return;
int fin = val[L][R][lim], pos = P[L][R][lim];
ans[pos] = fin;
findans(L, pos - , fin);
findans(pos + , R, fin);
} int main()
{
n = rd; m = rd;
for (int i = ; i <= m; ++i) {
a[i].l = rd; a[i].r = rd; a[i].val = rd;
ls[++tot] = a[i].val;
}
sort(ls + , ls + + tot);
tot = unique(ls + , ls + + tot) - ls - ;
for (int i = ; i <= m; ++i)
a[i].val = fd(a[i].val);
for (int i = n; i; --i)
for (int j = i; j <= n; ++j)
DP(i, j);
printf("%d\n", dp[][n][]);
findans(, n, );
for (int i = ; i <= n; ++i)
printf("%d ", ls[ans[i]]);
puts("");
}
BZOJ4380 Myjnie / Luogu3592 [POI2015]MYJ-区间DP的更多相关文章
- 2018.10.22 bzoj4380: [POI2015]Myjnie(区间dp)
传送门 区间dp好题. f[i][j][k]f[i][j][k]f[i][j][k]表示区间[i,j][i,j][i,j]最小值为kkk时的最大贡献. 然后可以枚举端点转移. 当时口胡到这儿就不会了. ...
- 【BZOJ-4380】Myjnie 区间DP
4380: [POI2015]Myjnie Time Limit: 40 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 162 Solved: ...
- 【BZOJ 4380】4380: [POI2015]Myjnie (区间DP)
4380: [POI2015]Myjnie Description 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i].有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗 ...
- BZOJ 4380 Myjnie 区间DP
4380: [POI2015]Myjnie Time Limit: 40 Sec Memory Limit: 256 MBSec Special JudgeSubmit: 162 Solved: ...
- P3592 [POI2015]MYJ
P3592 [POI2015]MYJ 一道比较烦的区间dp.. 昨天上课讲到了这题,然后就在lg翻到了 然后调了很久很久..... 设\(f[l][r][k]\)为区间\([l,r]\)中,最小值\( ...
- luogu P3592 [POI2015]MYJ
题目链接 luogu P3592 [POI2015]MYJ 题解 区间dp 设f[l][r][k]表示区间l到r内最小值>=k的最大收益 枚举为k的位置p,那么包含p的区间答案全部是k 设h[i ...
- 区间dp提升复习
区间\(dp\)提升复习 不得不说这波题真的不简单... 技巧总结: 1.有时候转移可以利用背包累和 2.如果遇到类似区间添加限制的题可以直接把限制扔在区间上,每次只考虑\([l,r]\)被\([i, ...
- 【POJ-1390】Blocks 区间DP
Blocks Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 5252 Accepted: 2165 Descriptio ...
- 区间DP LightOJ 1422 Halloween Costumes
http://lightoj.com/volume_showproblem.php?problem=1422 做的第一道区间DP的题目,试水. 参考解题报告: http://www.cnblogs.c ...
随机推荐
- redis 双写一致性 看一篇成高手系列1
首先,缓存由于其高并发和高性能的特性,已经在项目中被广泛使用.在读取缓存方面,大家没啥疑问,都是按照下图的流程来进行业务操作. 但是在更新缓存方面,对于更新完数据库,是更新缓存呢,还是删除缓存.又或者 ...
- SpringBoot项目单元测试
关于SpringBoot的单元测试,描述一下三种单元测试的方式. 1.约定 单元测试代码写在src/test/java目录下单元测试类命名为*Test,前缀为要测试的类名 2. 使用mock方式单元测 ...
- USB3.0及NVME SSD安装WIN7X64
USB3.0及NVME SSD安装WIN7X64https://tieba.baidu.com/p/4822034273?pn=1所有的人都是菜鸟过来的,不过有些人懂得自己动手找到答案:有些人则是懒得 ...
- 2、evaluate-reverse-polish-notation
题目描述 Evaluate the value of an arithmetic expression in Reverse Polish Notation. Valid operators are+ ...
- JDBC 中preparedStatement和Statement区别
一.概念 PreparedStatement是用来执行SQL查询语句的API之一,Java提供了 Statement.PreparedStatement 和 CallableStatement三种方式 ...
- .net mvc 分页
1.分页实体类 public class PageDto { public int PageIndex { get; set; } public int PageSize { get; set; } ...
- 第25课 可变参数模板(6)_function_traits和ScopeGuard的实现
1. function_traits (1)function_traits的作用:获取函数的实际类型.返回值类型.参数个数和具体类型等.它能获取所有函数语义类型信息.可以获取普通函数.函数指针.std ...
- suricata 配置文件threshold
threshold threshold(阈值)关键字可用于控制规则的警报频率,它有3种模式: threshold: type <threshold|limit|both>, track & ...
- android 开发 View _3_ View的属性动画ValueAnimator
ValueAnimator ValueAnimator继承自抽象类Animator.要让属性动画渐变式地更改对象中某个属性的值,可分两步操作:第一步,动画需要计算出某一时刻属性值应该是多少:第二步,需 ...
- 为什么SQL用UPDATE语句更新时更新行数会多3行有触发器有触发器有触发器有触发器有触发器有触发器
update明显更新就一行,但是结果显示更新多行. 原因是有触发器有触发器有触发器有触发器有触发器有触发器有触发器有触发器有触发器