简单dp,设$f_{i,j,k}$表示第i个时间段,钢琴处在(j,k)位置移动距离的最大值,那么有转移

$f_{i, j, k} = max(f_{i - 1, j, k}) ,  f_{i, j, k} = max(f_{i - 1, a, b})$

其中 

$j - dx_{d_{i}} * len\leq a \leq j -1 $

$k - dy_{d_{i}} * len\leq b \leq k -1 $
$len = ed_{i} - st_{i} + 1 $

其实就是一个滑动窗口,考虑到转移顺序,就是写四遍单调队列~
但是我成功地把方程写错了…… i - 1 写成了 i ,改掉就A了

Code:

#include <cstdio>
#include <cstring>
#include <cstdlib>
using namespace std; const int N = ; int n, m, segNum, st[N], ed[N], d[N];
int fx, fy, f[N][N][N], q[N];
char mp[N][N]; inline void read(int &X) {
X = ;
char ch = ;
int op = ;
for(; ch > ''|| ch < ''; ch = getchar())
if(ch == '-') op = -;
for(; ch >= '' && ch <= ''; ch = getchar())
X = (X << ) + (X << ) + ch - ;
X *= op;
} inline void chkMax(int &x, int y) {
if(y > x) x = y;
} inline void print(int p) {
printf("\n%d\n", p);
for(int i = ; i <= n; i++, printf("\n"))
for(int j = ; j <= m; j++)
printf("%10d ", f[p][i][j]);
system("pause");
} int main() {
read(n), read(m), read(fx), read(fy), read(segNum);
for(int i = ; i <= n; i++) scanf("%s", mp[i] + );
for(int i = ; i <= segNum; i++)
read(st[i]), read(ed[i]), read(d[i]); memset(f, 0xcf, sizeof(f));
f[][fx][fy] = ;
for(int i = ; i <= segNum; i++) {
for(int j = ; j <= n; j++)
for(int k = ; k <= m; k++)
if(mp[j][k] != 'x') chkMax(f[i][j][k], f[i - ][j][k]); // print(i); int len = ed[i] - st[i] + ;
if(d[i] == ) {
for(int k = ; k <= m; k++) {
int l = , r = ;
for(int j = n; j >= ; j--) {
for(; l <= r && q[l] > j + len; l++);
if(mp[j][k] == 'x') l = , r = ;
else if(l <= r) chkMax(f[i][j][k], f[i - ][q[l]][k] + (q[l] - j));
if(f[i - ][j][k] >= ) {
for(; l <= r && f[i - ][q[r]][k] + (q[r] - j) < f[i - ][j][k]; r--);
q[++r] = j;
}
}
}
} if(d[i] == ) {
for(int k = ; k <= m; k++) {
int l = , r = ;
for(int j = ; j <= n; j++) {
for(; l <= r && q[l] < j - len; l++);
if(mp[j][k] == 'x') l = , r = ;
else if(l <= r) chkMax(f[i][j][k], f[i - ][q[l]][k] + (j - q[l]));
if(f[i - ][j][k] >= ) {
for(; l <= r && f[i - ][q[r]][k] + (j - q[r]) < f[i - ][j][k]; r--);
q[++r] = j;
}
}
}
} if(d[i] == ) {
for(int j = ; j <= n; j++) {
int l = , r = ;
for(int k = m; k >= ; k--) {
for(; l <= r && q[l] > k + len; l++);
if(mp[j][k] == 'x') l = , r = ;
else if(l <= r) chkMax(f[i][j][k], f[i - ][j][q[l]] + (q[l] - k));
if(f[i - ][j][k] >= ) {
for(; l <= r && f[i - ][j][q[r]] + (q[r] - k) < f[i - ][j][k]; r--);
q[++r] = k;
}
}
}
} if(d[i] == ) {
for(int j = ; j <= n; j++) {
int l = , r = ;
for(int k = ; k <= m; k++) {
for(; l <= r && q[l] < k - len; l++);
if(mp[j][k] == 'x') l = , r = ;
else if(l <= r) chkMax(f[i][j][k], f[i - ][j][q[l]] + (k - q[l]));
if(f[i - ][j][k] >= ) {
for(; l <= r && f[i - ][j][q[r]] + (k - q[r]) < f[i - ][j][k]; r--);
q[++r] = k;
}
}
}
} // print(i);
} int ans = ;
for(int i = ; i <= n; i++)
for(int j = ; j <= m; j++)
chkMax(ans, f[segNum][i][j]);
printf("%d\n", ans); return ;
}

还是要仔细a

Luogu 2254 [NOI2005]瑰丽华尔兹的更多相关文章

  1. luogu P2254 [NOI2005]瑰丽华尔兹

    题目链接 luogu P2254 [NOI2005]瑰丽华尔兹 题解 为什么我我我不放放放bzoj的链接呢? 因为打的暴力啊,然后bzojT了呀QAQQQQQ(逃 然后luogu竟然过了呀呀呀 dp[ ...

  2. 单调队列优化DP || [NOI2005]瑰丽华尔兹 || BZOJ 1499 || Luogu P2254

    题外话:题目极好,做题体验极差 题面:[NOI2005]瑰丽华尔兹 题解: F[t][i][j]表示第t时刻钢琴位于(i,j)时的最大路程F[t][i][j]=max(F[t-1][i][j],F[t ...

  3. NOI2005瑰丽华尔兹

    1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 893  Solved: 508[Submit][Status] ...

  4. bzoj1499[NOI2005]瑰丽华尔兹 单调队列优化dp

    1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 1802  Solved: 1097[Submit][Status ...

  5. 【BZOJ1499】[NOI2005]瑰丽华尔兹 单调队列+DP

    [BZOJ1499][NOI2005]瑰丽华尔兹 Description 你跳过华尔兹吗?当音乐响起,当你随着旋律滑动舞步,是不是有一种漫步仙境的惬意?众所周知,跳华尔兹时,最重要的是有好的音乐.但是 ...

  6. [Bzoj1499][NOI2005]瑰丽华尔兹[简单DP]

    1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec  Memory Limit: 64 MBSubmit: 1714  Solved: 1042[Submit][Status ...

  7. P2254 [NOI2005]瑰丽华尔兹

    链接P2254 [NOI2005]瑰丽华尔兹 首先有个很朴素的\(dp\),设\(f_{i,j,k}\)表示\(k\)时刻地点\(i,j\)的最长长度. 然后这样不能优化,考虑利用一段连续时间是同一个 ...

  8. 题解-[NOI2005]瑰丽华尔兹

    题解-[NOI2005]瑰丽华尔兹 [NOI2005]瑰丽华尔兹 \(n\times m\) 的矩阵.以 \((x,y)\) 为起点.一共 \(k\) 段时间,每段时间为 \([s_i,t_i](t_ ...

  9. BZOJ 1499 [NOI2005] 瑰丽华尔兹 | 单调队列优化DP

    BZOJ 1499 瑰丽华尔兹 | 单调队列优化DP 题意 有一块\(n \times m\)的矩形地面,上面有一些障碍(用'#'表示),其余的是空地(用'.'表示).每时每刻,地面都会向某个方向倾斜 ...

随机推荐

  1. DataTable的用法

    在项目中经常用到DataTable,如果DataTable使用得当,不仅能使程序简洁实用,而且能够提高性能,达到事半功倍的效果,现对DataTable的使用技巧进行一下总结. 一.DataTable简 ...

  2. uva11729 - Commando War(贪心)

    贪心法,执行任务的时间J越长的应该越先交待.可以用相邻交换法证明正确性.其实对于两个人,要让总时间最短,就要让同一时间干两件事的时间最长. #include<iostream> #incl ...

  3. redis3.2.11 安装

    wget http://download.redis.io/releases/redis-3.2.11.tar.gz [root@hdp01 src]# .tar.gz -C /opt/ [root@ ...

  4. C#面向对象(四):其他面向对象知识

    前文链接: C#面向对象(一):明确几个简单的概念作为开胃菜 C#面向对象(二):封装和继承 C#面向对象(三):多态 今天是这个系列的收尾文章了,来谈谈其他面向对象知识. 1.嵌套类 1.1概念 在 ...

  5. mysql时间随笔

    SELECT FROM_UNIXTIME(create_time,'%Y-%m-%d %H:%i:%s') FROM `order`; select date_add(FROM_UNIXTIME(cr ...

  6. python函数之sorted与sort

    Python list内置sort()方法用来排序,也可以用python内置的全局sorted()方法来对可迭代的序列排序生成新的序列. sorted(iterable,key=None,revers ...

  7. 分区工具parted的详解及常用分区使用方法

    一.         parted的用途及说明 概括使用说明: parted用于对磁盘(或RAID磁盘)进行分区及管理,与fdisk分区工具相比,支持2TB以上的磁盘分区,并且允许调整分区的大小.   ...

  8. Linux系统下如何去掉文件的@属性

    前几天从别处copy了一个memcached.so文件,发现运行不了,用ls -l 看了一下发现memcached.so多了一个@属性,如何去掉这个@属性呢? 第一步:ls -laeO@ 第二步:xa ...

  9. STM32 -- 故障记录

    1.串口2无法发送数据 1)串口2和串口1使用的时钟总线不同: usart1:RCC_APBPeriphClockCmd(RCC_APB2Periph_USART1,ENABLE); usart2:R ...

  10. HTTP-Runoob:HTTP简介

    ylbtech-HTTP-Runoob:HTTP简介 1.返回顶部 1. HTTP 简介 HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网 ...