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 ...
随机推荐
- Linux中redis安装配置及使用详解
Linux中redis安装配置及使用详解 一. Redis基本知识 1.Redis 的数据类型 字符串 , 列表 (lists) , 集合 (sets) , 有序集合 (sorts sets) , 哈 ...
- sequelize的get/post方法例子
定义两个model,一个给get的,一个给post的 var Sequelize = require('sequelize'); const DeviceNos = sequelize.define( ...
- 【spring】之xml和Annotation,Bean注入的方式
基于xml形式Bean注入 @Data @AllArgsConstructor @NoArgsConstructor public class PersonBean { private Integer ...
- Python【每日一问】02
问:列表 test = [1,2,3,1,3,4,5,67,7,8,54,1,2,3,4,5,6],如何删除该列表的重复元素? 方法1:利用集合的不重复性 # 利用集合的不重复性 test = [1, ...
- [UE4]Border
Border: 边界; 边; 镶边; 包边; Border也是一个容器,只能包含一个子元素. 一.添加一个名为testBorder的UserWidget,添加一个Border到默认成Canvas Pa ...
- [UE4]Retainer Box
把子元素的内容渲染到一个Render Target上去,然后放把它放置到到屏幕上去. Retainer Box的作用: 1.控制UI更新频率 2.把渲染后的UI当成Texture,放入材质中,加工后, ...
- 实验一:通过bridge-utils工具创建网桥并实现网络连接
实验名称: 通过bridge-utils工具创建网桥并实现网络连接 实验环境: 实验要求: 安装bridge-utils工具,创建网桥br0,通过brctl命令,为网桥配置IP地址192.168.23 ...
- html 设置input框的记忆功能(联想内容)
autocomplete=“on/off” 1.默认情况下,autocomplete的值是on.你可以将其设置为off. 2.autocomplete属性可以放在input 元素上,也可以放在form ...
- (转)适用微信小程序的table表格(带隔行变色)
原文地址 table.wxml <view class="table"> <view class="tr bg-w"> <view ...
- C# 操作符 << 与 >>
1.<< 左移操作符: 左移操作符,将第一个操作数向左移动第二个操作数指定的位数,空出的位置补0.左移相当于乘. 左移一位相当于乘2;左移两位相当于乘4;左移三位相当于乘8. 如:x< ...