@description@

有 n 家洗车店从左往右排成一排,每家店都有一个正整数价格 p[i]。

有 m 个人要来消费,第 i 个人会驶过第 a[i] 个开始一直到第 b[i] 个洗车店,且会选择这些店中最便宜的一个进行一次消费。但是如果这个最便宜的价格大于 c[i],那么这个人就不洗车了。

请给每家店指定一个价格,使得所有人花的钱的总和最大。

input

第一行包含两个正整数 n, m(1<=n<=50,1<=m<=4000)。

接下来 m 行,每行包含三个正整数 a[i], b[i], ci

output

第一行输出一个正整数,即消费总额的最大值。

第二行输出 n 个正整数,依次表示每家洗车店的价格 p[i],要求 1<=p[i]<=500000。

若有多组最优解,输出任意一组。

sample input

7 5

1 4 7

3 7 13

5 6 20

6 7 1

1 2 5

sample output

43

5 5 13 13 20 20 13

@solution@

首先,洗车店的价格一定是某顾客的 c 值,否则可以通过调整变得更优。

像这样的区间最值相关问题可以考虑其笛卡尔树的结构。比如这道题就可以用笛卡尔树来 dp。

具体来讲,我们定义状态 dp(i, j, k),表示区间 [i, j] 的最小值为 k 能够取得的最大利益。根据最先的结论,我们 k 的取值只有 O(m) 个。

通过枚举 [i, j] 的最小值位置 p,我们可以进行转移:

\[dp(i, j, k) = dp(i, p-1, x) + dp(p+1, j, y) + f(i, j, p, k)
\]

其中要求 x, y >= k,权值函数 f(i, j, p, k) 表示这一个状态能够赚得的顾客利益。

f(i, j, p, k) 会计算到哪些顾客呢?假如第 q 个顾客被计算到了,那么必然有 i <= a[q] <= p <= b[q] <= j 且 c[q] >= k。

通过从大到小枚举 k,可以快速地维护 f。

我们通过维护后缀最大值,就可以不必再枚举 x 和 y。

因为要输出方案,记得记录当前状态是怎么转移过来的。最后递归输出即可。

总时间复杂度 O(n^3*m)。

@accepted code@

#include<cstdio>
#include<algorithm>
using namespace std;
const int MAXN = 50;
const int MAXM = 4000;
struct node{
int le, ri;
int lim;
}a[MAXM + 5];
bool operator < (node a, node b) {
return a.lim < b.lim;
}
int n, m;
int ans[MAXN + 5];
int mx[MAXN + 5][MAXN + 5][MAXM + 5];
int dp[MAXN + 5][MAXN + 5][MAXM + 5];
int val[MAXN + 5][MAXN + 5][MAXN + 5];
int pre[MAXN + 5][MAXN + 5][MAXM + 5];
int mxpos[MAXN + 5][MAXN + 5][MAXM + 5];
void GetAnswer(int le, int ri, int p) {
if( mx[le][ri][p] == 0 ) {
for(int i=le;i<=ri;i++)
ans[i] = a[p].lim;
return ;
}
int q = mxpos[le][ri][p];
int r = pre[le][ri][q];
ans[r] = a[q].lim;
GetAnswer(le, r-1, q);
GetAnswer(r+1, ri, q);
}
int main() {
scanf("%d%d", &n, &m);
for(int i=1;i<=m;i++)
scanf("%d%d%d", &a[i].le, &a[i].ri, &a[i].lim);
sort(a+1, a+m+1);
for(int len=1;len<=n;len++)
for(int i=1;i+len-1<=n;i++) {
int j = i + len - 1;
for(int k=m;k>=1;k--) {
if( i <= a[k].le && a[k].ri <= j ) {
for(int l=i;l<=j;l++) {
if( a[k].le <= l && l <= a[k].ri )
val[i][j][l]++;
if( mx[i][l-1][k] + mx[l+1][j][k] + val[i][j][l]*a[k].lim > dp[i][j][k] ) {
pre[i][j][k] = l;
dp[i][j][k] = mx[i][l-1][k] + mx[l+1][j][k] + val[i][j][l]*a[k].lim;
}
}
}
if( k == m || dp[i][j][k] >= mx[i][j][k+1] ) {
mx[i][j][k] = dp[i][j][k];
mxpos[i][j][k] = k;
}
else {
mx[i][j][k] = mx[i][j][k+1];
mxpos[i][j][k] = mxpos[i][j][k+1];
}
}
}
printf("%d\n", mx[1][n][1]);
GetAnswer(1, n, 1);
for(int i=1;i<=n;i++)
printf("%d ", ans[i]);
puts("");
}

@details@

其实不需要离散化,因为即使 c 值重复也影响不大。

一开始因为区间 dp 的某些错误 dp 方式 WA 了几次(捂脸)。

@bzoj - 4380@ [POI2015] Myjnie的更多相关文章

  1. BZOJ 4380 [POI2015]Myjnie | DP

    链接 BZOJ 4380 题面 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i]. 有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗车店,且会选择这些店中最便宜的一个 ...

  2. 【BZOJ 4380】4380: [POI2015]Myjnie (区间DP)

    4380: [POI2015]Myjnie Description 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i].有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗 ...

  3. 【BZOJ】4380: [POI2015]Myjnie

    题解 区间dp,先离散化所有价值 \(f[i][j][k]\)表示\([i,j]\)区间里最小值为\(k\)的价值最大是多少 只考虑\(i <= a <= b <= j\)的区间,枚 ...

  4. BZOJ 4380 Myjnie 区间DP

    4380: [POI2015]Myjnie Time Limit: 40 Sec  Memory Limit: 256 MBSec  Special JudgeSubmit: 162  Solved: ...

  5. bzoj [POI2015]Myjnie

    [POI2015]Myjnie Time Limit: 40 Sec Memory Limit: 256 MBSec Special Judge Description 有n家洗车店从左往右排成一排, ...

  6. [POI2015]Myjnie

    [POI2015]Myjnie 题目大意: 有\(n(n\le50)\)家洗车店从左往右排成一排,每家店都有一个正整数价格\(d_i\). 有\(m(m\le4000)\)个人要来消费,第\(i\)个 ...

  7. 【BZOJ4380】[POI2015]Myjnie 区间DP

    [BZOJ4380][POI2015]Myjnie Description 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i].有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[ ...

  8. bzoj4380[POI2015]Myjnie dp

    [POI2015]Myjnie Time Limit: 40 Sec  Memory Limit: 256 MBSec  Special JudgeSubmit: 368  Solved: 185[S ...

  9. bzoj 4386: [POI2015]Wycieczki

    bzoj 4386: [POI2015]Wycieczki 这题什么素质,爆long long就算了,连int128都爆……最后还是用long double卡过的……而且可能是我本身自带大常数吧,T了 ...

随机推荐

  1. dom4解析 XML

    Java XML解析工具 dom4j介绍及使用实例 Java XML解析工具 dom4j介绍及使用实例 dom4j介绍 dom4j的项目地址:http://sourceforge.net/projec ...

  2. sar网络统计数据

    sar是一个研究磁盘I/O的优秀工具.以下是sar磁盘I/O输出的一个示例. 第一行-d显示磁盘I/O信息,5 2选项是间隔和迭代,就像sar数据收集器那样.表3-3列出了字段和说明. 表3-3    ...

  3. 考试总结 模拟28(W)

    心得: 状态极差,都怪放假,上一套的T3没改完,今天考试没有一点状态,开学恐惧症.(不恐惧作业或一调但还是很茫然) T1能A掉实在是意外,杂题T1没做过,可能是人品守恒,(丢了钱今天才发现以后一定锁柜 ...

  4. LUOGU P2921 [USACO08DEC]在农场万圣节Trick or Treat on the Farm

    传送门 解题思路 记忆化搜索,如果搜到环,就将环的大小处理出来. 代码 #include<iostream> #include<cstdio> #include<cstr ...

  5. 【JZOJ5064】【GDOI2017第二轮模拟day2】友好城市 Kosarajo算法+bitset+ST表+分块

    题面 在Byteland 一共有n 座城市,编号依次为1 到n,这些城市之间通过m 条单向公路连接. 对于两座不同的城市a 和b,如果a 能通过这些单向道路直接或间接到达b,且b 也能如此到达a,那么 ...

  6. 蚁群算法MATLAB解VRP问题

    Excel  exp12_3_2.xls内容: ANT_VRP函数: function [R_best,L_best,L_ave,Shortest_Route,Shortest_Length]=ANT ...

  7. spring boot 异常处理(转)

    spring boot在异常的处理中,默认实现了一个EmbeddedServletContainerCustomizer并定义了一个错误页面到"/error"中,在ErrorMvc ...

  8. BZOJ 3057圣主的考验题解

    老师居然考这么毒瘤的题目!!!!! 很容易想到dp,f[i][j]表示有i个节点,左子树的最深深度为j的方案数 枚举左子树有多少节点然后转移,复杂度为n^3 T飞~ 我们考虑到有深度为h的树的节点有多 ...

  9. 部署 LAMP (CentOS 7.2)

    摘自:https://help.aliyun.com/document_detail/50774.html 简介 LAMP指Linux+Apache+Mysql/MariaDB+Perl/PHP/Py ...

  10. PHP-xdebug+PHPStorm的debug安装(未完)

    PHP的xdebug安装 一.首先去phpinfo()查看自己的php是32还是64位的,再根据php版本在下面的网站https://xdebug.org/download.php选择xdebug插件 ...