BZOJ 1499 NOI2005 瑰丽华尔兹 单调队列
题目大意:给定一个m*n的地图,一些点有障碍物,钢琴初始在一个点,每一个时间段能够选择向给定的方向移动一段距离,求最长路径长
朴素DP的话,我们有T个时间段,每一个时间段有m*n个点,n个时间,一定会超时
考虑到一个时间段全部的更新操作都是同样的,我们能够考虑单调队列优化
设队尾为(x,y),新插入的点为(x',y'),那么当Distance( (x,y) , (x',y') ) <= f[x'][y'] - f[x][y]时,(x,y)可被删掉
四遍单调队列就可以 O(T*m*n)
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define M 210
using namespace std;
typedef pair<int,int> abcd;
int n,m,k,ans;
char map[M][M];
int f[M][M],g[M][M];
abcd q[M];int r,h;
inline int Distance(const abcd &x,const abcd &y)
{
return abs(x.first-y.first)+abs(x.second-y.second);
}
inline void Insert(const abcd &x)
{
while( r!=h && Distance(x,q[r]) <= f[x.first][x.second] - f[q[r].first][q[r].second] )
--r;
q[++r]=x;
}
inline int Get_Ans(const abcd &x,int len)
{
while( r!=h && Distance(q[h+1],x)>len )
++h;
if(r==h)
return 0xefefefef;
return f[q[h+1].first][q[h+1].second]+Distance(q[h+1],x);
}
void U(int len)
{
int i,j;
for(j=1;j<=n;j++)
{
r=h=0;
for(i=m;i;i--)
if(map[i][j]=='.')
{
abcd p(i,j);
g[i][j]=max( f[i][j] , Get_Ans(p,len) );
Insert(p);
}
else
r=h=0;
}
memcpy(f,g,sizeof f);
}
void D(int len)
{
int i,j;
for(j=1;j<=n;j++)
{
r=h=0;
for(i=1;i<=m;i++)
if(map[i][j]=='.')
{
abcd p(i,j);
g[i][j]=max( f[i][j] , Get_Ans(p,len) );
Insert(p);
}
else
r=h=0;
}
memcpy(f,g,sizeof f);
}
void L(int len)
{
int i,j;
for(i=1;i<=m;i++)
{
r=h=0;
for(j=n;j;j--)
if(map[i][j]=='.')
{
abcd p(i,j);
g[i][j]=max( f[i][j] , Get_Ans(p,len) );
Insert(p);
}
else
r=h=0;
}
memcpy(f,g,sizeof f);
}
void R(int len)
{
int i,j;
for(i=1;i<=m;i++)
{
r=h=0;
for(j=1;j<=n;j++)
if(map[i][j]=='.')
{
abcd p(i,j);
g[i][j]=max( f[i][j] , Get_Ans(p,len) );
Insert(p);
}
else
r=h=0;
}
memcpy(f,g,sizeof f);
}
int main()
{
int i,j,x,y,z;
cin>>m>>n>>x>>y>>k;
for(i=1;i<=m;i++)
scanf("%s",map[i]+1);
memset(f,0xef,sizeof f);
memset(g,0xef,sizeof g);
f[x][y]=0;
for(i=1;i<=k;i++)
{
scanf("%d%d%d",&x,&y,&z);
switch(z)
{
case 1:U(y-x+1);break;
case 2:D(y-x+1);break;
case 3:L(y-x+1);break;
case 4:R(y-x+1);break;
}
}
for(i=1;i<=m;i++)
for(j=1;j<=n;j++)
ans=max(ans,f[i][j]);
cout<<ans<<endl;
}
BZOJ 1499 NOI2005 瑰丽华尔兹 单调队列的更多相关文章
- BZOJ 1499 [NOI2005] 瑰丽华尔兹 | 单调队列优化DP
BZOJ 1499 瑰丽华尔兹 | 单调队列优化DP 题意 有一块\(n \times m\)的矩形地面,上面有一些障碍(用'#'表示),其余的是空地(用'.'表示).每时每刻,地面都会向某个方向倾斜 ...
- bzoj 1499 [NOI2005]瑰丽华尔兹——单调队列优化dp
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1499 简单的单调队列优化dp.(然而当时却WA得不行.今天总算填了坑) 注意滚动数组赋初值应 ...
- bzoj1499[NOI2005]瑰丽华尔兹 单调队列优化dp
1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 1802 Solved: 1097[Submit][Status ...
- 【BZOJ1499】[NOI2005]瑰丽华尔兹 单调队列+DP
[BZOJ1499][NOI2005]瑰丽华尔兹 Description 你跳过华尔兹吗?当音乐响起,当你随着旋律滑动舞步,是不是有一种漫步仙境的惬意?众所周知,跳华尔兹时,最重要的是有好的音乐.但是 ...
- ●BZOJ 1499 [NOI2005]瑰丽华尔兹
题链: http://www.lydsy.com/JudgeOnline/problem.php?id=1499 题解: 单调队列优化DP 定义 dp[t][x][y] 表示第t个时间段之后,处在(x ...
- bzoj1499 [NOI2005]瑰丽华尔兹——单调队列优化DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1499 朴素DP方程很好想,以右移为例,就是 f[i][x][y]=max(f[i][x][y ...
- bzoj 1499: [NOI2005]瑰丽华尔兹【dp+单调队列】
设f[a][i][j]为第a段时间结束时在(i,j)位置的最长滑行距离,转移很好想,就是分四个方向讨论,然后枚举这段时间的滑行长度取个max即可 但是这样是O(n^4)的,考虑优化 发现同一行或列,取 ...
- NOI2005瑰丽华尔兹
1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 893 Solved: 508[Submit][Status] ...
- [Bzoj1499][NOI2005]瑰丽华尔兹[简单DP]
1499: [NOI2005]瑰丽华尔兹 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 1714 Solved: 1042[Submit][Status ...
随机推荐
- BZOJ 3385: [Usaco2004 Nov]Lake Counting 数池塘
题目 3385: [Usaco2004 Nov]Lake Counting 数池塘 Time Limit: 1 Sec Memory Limit: 128 MB Description 农夫 ...
- Book of Evil 树双向DFS
Book of Evil Paladin Manao caught the trail of the ancient Book of Evil in a swampy area. This area ...
- 过河(bfs)
Problem 2188 过河I Accept: 112 Submit: 277 Time Limit: 3000 mSec Memory Limit : 32768 KB Proble ...
- JS实现日历控件选择后自动填充
最近在做人事档案的项目,在做项目的初期对B/S这块不是很熟悉,感觉信心不是很强,随着和师哥同组人员的交流后发现,调试程序越来越好了,现在信心是倍增,只要自己自己踏实的去研究.理解代码慢慢的效果就出来了 ...
- virtual 关键字
virtual 关键字用于修饰方法.属性.索引器或事件声明,并且允许在派生类中重写这些对象.例如,此方法可被任何继承它的类重写. public virtual double Area() { retu ...
- MVC表单提交加JS验证
做一个普通表单提交,但是要加前端验证,如下: 1. Action public ActionResult Add(ProductModelproductID) { //operate... } ...
- BZOJ 1202: [HNOI2005]狡猾的商人( 差分约束 )
好像很多人用并查集写的... 前缀和, 则 sumt - sums-1 = v, 拆成2条 : sumt ≤ sums-1 + v, sums-1 ≤ sumt - v 就是一个差分约束, 建图跑SP ...
- C++创建对象的三种方式
C++在创建对象的时候,有三种方式: #include <iostream> using namespace std; class A { private: int n; public: ...
- CURL采集
<?php $url='';//输入'网址 $ch = curl_init(); $timeout = 5;//超时时间 curl_setopt ($ch, CURLOPT_URL, $url) ...
- java-StringTokenizer类
StringTokenizer类别可以通过某个字符或者多个字符作为分界符,来将字符串划分为多个标记(token). package com.example.helloworld; import jav ...