BZOJ1114 : [POI2008]鲁滨逊逃生Rob
设船最宽行列的交点为船的重心,那么只要预处理出重心在每个位置是否可行,以及在边界上走出边界所需的最小值之后,进行一遍BFS即可。
枚举每个点$(x,y)$,求出它上下最近的障碍物的距离。考虑重心在第$x$行左右移动的时候,不能走的部分是一个区间,可以$O(1)$计算。
对于左右障碍物也同理,因此可以通过维护差分前缀和可以做到$O(n^2)$预处理出所有可行位置。
对于边界,首先一定可以通过一直往一个方向走得到一个解;其次对于四个角落上的点,可以通过$O(n)$枚举它卡住船的哪个角落来得到走出地图的最优解。
总时间复杂度为$O(n^2)$。
#include<cstdio>
const int N=2010,inf=100000000;
int n,i,j,x,y,z,xl=inf,xr,yl=inf,yr,xm,ym,d,mu,md,l,r;
int f[N][N],h,t,q[N*N][2],ans=inf;
int ul[N],ur[N],dl[N],dr[N],fu[N],fd[N],f11,f1n,fn1,fnn;
char a[N][N],ban[N][N];
inline void umin(int&a,int b){if(a>b)a=b;}
inline void umax(int&a,int b){if(a<b)a=b;}
inline void add(int x,int y,int z){
if(x<1||x>n||y<1||y>n)return;
if(ban[x][y]||~f[x][y])return;
if(x==1)umin(ans,z+xr-xm+1);
if(x==n)umin(ans,z+xm-xl+1);
if(y==1)umin(ans,z+yr-ym+1);
if(y==n)umin(ans,z+ym-yl+1);
if(x==1&&y==1)umin(ans,z+f11);
if(x==1&&y==n)umin(ans,z+f1n);
if(x==n&&y==1)umin(ans,z+fn1);
if(x==n&&y==n)umin(ans,z+fnn);
q[++t][0]=x,q[t][1]=y,f[x][y]=z;
}
int main(){
scanf("%d",&n);
for(i=1;i<=n;i++)for(scanf("%s",a[i]+1),j=1;j<=n;j++)if(a[i][j]=='r'){
umin(xl,i);
umax(xr,i);
umin(yl,j);
umax(yr,j);
}
for(i=xl;i<=xr;i++){
for(j=yl;j<=yr;j++)if(a[i][j]!='r')break;
if(j>yr)xm=i;
}
for(i=yl;i<=yr;i++){
for(j=xl;j<=xr;j++)if(a[j][i]!='r')break;
if(j>xr)ym=i;
}
for(i=xl;i<=xm;i++){
d=xm-i;
ul[d]=inf;ur[d]=0;
for(j=yl;j<=yr;j++)if(a[i][j]=='r'){
umin(ul[d],j);
umax(ur[d],j);
}
ul[d]=ym-ul[d];
ur[d]=ur[d]-ym;
}
for(i=xm;i<=xr;i++){
d=i-xm;
dl[d]=inf;dr[d]=0;
for(j=yl;j<=yr;j++)if(a[i][j]=='r'){
umin(dl[d],j);
umax(dr[d],j);
}
dl[d]=ym-dl[d];
dr[d]=dr[d]-ym;
}
mu=xm-xl;
md=xr-xm;
for(i=1;i<=n;i++)for(j=1;j<=n;j++)f[i][j]=0;
for(i=1;i<=n;i++){
fu[0]=fd[n+1]=inf;
for(j=1;j<=n;j++)if(a[j][i]=='X')fu[j]=0;else fu[j]=fu[j-1]+1;
for(j=n;j;j--)if(a[j][i]=='X')fd[j]=0;else fd[j]=fd[j+1]+1;
for(j=1;j<=n;j++){
l=n;
if(fu[j]<=mu)umin(l,i-ur[fu[j]]);
if(fd[j]<=md)umin(l,i-dr[fd[j]]);
r=1;
if(fu[j]<=mu)umax(r,i+ul[fu[j]]);
if(fd[j]<=md)umax(r,i+dl[fd[j]]);
umax(l,1),umin(r,n);
if(l<=r)f[j][l]++,f[j][r+1]--;
}
}
for(i=1;i<=n;i++)for(j=1;j<=n;j++){
f[i][j]+=f[i][j-1];
if(f[i][j])ban[i][j]=1;
}
f11=f1n=fn1=fnn=inf;
for(i=1;i<=md;i++){
umin(f11,i+dr[i]+1);
umin(f1n,i+dl[i]+1);
umin(fn1,i+ur[i]+1);
umin(fnn,i+ul[i]+1);
}
for(i=yl;i<=ym;i++){
d=ym-i;
ul[d]=inf;ur[d]=0;
for(j=xl;j<=xr;j++)if(a[j][i]=='r'){
umin(ul[d],j);
umax(ur[d],j);
}
ul[d]=xm-ul[d];
ur[d]=ur[d]-xm;
}
for(i=ym;i<=yr;i++){
d=i-ym;
dl[d]=inf;dr[d]=0;
for(j=xl;j<=xr;j++)if(a[j][i]=='r'){
umin(dl[d],j);
umax(dr[d],j);
}
dl[d]=xm-dl[d];
dr[d]=dr[d]-xm;
}
mu=ym-yl;
md=yr-ym;
for(i=1;i<=n;i++)for(j=1;j<=n;j++)f[i][j]=0;
for(i=1;i<=n;i++){
fu[0]=fd[n+1]=inf;
for(j=1;j<=n;j++)if(a[i][j]=='X')fu[j]=0;else fu[j]=fu[j-1]+1;
for(j=n;j;j--)if(a[i][j]=='X')fd[j]=0;else fd[j]=fd[j+1]+1;
for(j=1;j<=n;j++){
l=n;
if(fu[j]<=mu)umin(l,i-ur[fu[j]]);
if(fd[j]<=md)umin(l,i-dr[fd[j]]);
r=1;
if(fu[j]<=mu)umax(r,i+ul[fu[j]]);
if(fd[j]<=md)umax(r,i+dl[fd[j]]);
umax(l,1),umin(r,n);
if(l<=r)f[l][j]++,f[r+1][j]--;
}
}
for(i=1;i<=n;i++)for(j=1;j<=n;j++){
f[i][j]+=f[i-1][j];
if(f[i][j])ban[i][j]=1;
}
for(h=i=1;i<=n;i++)for(j=1;j<=n;j++)f[i][j]=-1;
add(xm,ym,0);
while(h<=t){
x=q[h][0],y=q[h++][1],z=f[x][y]+1;
add(x+1,y,z);
add(x-1,y,z);
add(x,y+1,z);
add(x,y-1,z);
}
if(ans<inf)printf("%d",ans);else puts("NIE");
return 0;
}
BZOJ1114 : [POI2008]鲁滨逊逃生Rob的更多相关文章
- bzoj AC倒序
Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...
- [BZOJ1112][POI2008]砖块Klo
[BZOJ1112][POI2008]砖块Klo 试题描述 N柱砖,希望有连续K柱的高度是一样的. 你可以选择以下两个动作 1:从某柱砖的顶端拿一块砖出来,丢掉不要了. 2:从仓库中拿出一块砖,放到另 ...
- [bzoj1122][POI2008]账本BBB
1122: [POI2008]账本BBB Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 402 Solved: 202[Submit][Status ...
- HDU 4857 逃生 (反向拓扑排序 & 容器实现)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4857 逃生 Time Limit: 2000/1000 MS (Java/Others) Mem ...
- 谷歌大牛 Rob Pike 的 5 个编程原则
谷歌大牛 Rob Pike 的 5 个编程原则 简介: Rob Pike,目前谷歌公司最著名的软件工程师之一,曾是贝尔实验室Unix开发团队成员,Plan9操作系统开发的主要领导人,Inferno操作 ...
- BZOJ 1113: [Poi2008]海报PLA
1113: [Poi2008]海报PLA Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1025 Solved: 679[Submit][Statu ...
- BZOJ 1116: [POI2008]CLO
1116: [POI2008]CLO Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 922 Solved: 514[Submit][Status][ ...
- BZOJ 1112: [POI2008]砖块Klo
1112: [POI2008]砖块Klo Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1736 Solved: 606[Submit][Statu ...
- BZOJ 1124: [POI2008]枪战Maf
1124: [POI2008]枪战Maf Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 617 Solved: 236[Submit][Status ...
随机推荐
- 昨天在公司加班,上午好像就是弄一个ftp的linux服务问题
在网上找了一些方法,可是其中有通过匿名方式登陆,但是在root的权限下才能存放文件,可是把匿名用户登陆取消之后又不能登陆,就是没有列出怎么来添加一个ftp的用户,今天打算直接装一个linux系统在虚拟 ...
- 22.访问者模式(Vistor Pattern)
using System; using System.Collections; namespace ConsoleApplication5 { /// <summary> /// 访问者模 ...
- [LeetCode] Remove Duplicates from Sorted Array
Given a sorted array, remove the duplicates in place such that each element appear only once and ret ...
- SPOJ220 Relevant Phrases of Annihilation(后缀数组)
引用罗穗骞论文中的话: 先将n 个字符串连起来,中间用不相同的且没有出现在字符串中的字符隔开,求后缀数组.然后二分答案,再将后缀分组.判断的时候,要看是否有一组后缀在每个原来的字符串中至少出现两次,并 ...
- HDU 1227 Fast Food
题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=1227 题意:一维坐标上有n个点,位置已知,选出k(k <= n)个点,使得所有n个点与选定的点中 ...
- 攻城狮在路上(叁)Linux(二十四)--- linux设置开机挂载及镜像文件挂载
虽然可以手动进行文件系统的挂载,但是每次都手动挂载就会很麻烦,开机挂载的目的就是实现文件系统的自动挂载. 一.开机挂载:/etc/fstab及/etc/mtab 主要是通过修改/etc/fstab文件 ...
- WPF之MVVM(Step2)——自己实现DelegateCommand:ICommand
在自己实现MVVM时,上一篇的实现方式基本是不用,因其对于命令的处理不够方便,没写一个命令都需要另加一个Command的类.此篇主要介绍DelegateCommand来解决上面所遇到的问题. 首先,我 ...
- 笔记本电脑关闭小键盘(即打字按P出现星号键)
开关方法:Fn + NumLk (联想电脑的NumLk 一般为F8,其他电脑自己在键盘找找罗)
- Iphone [Tab Bar实现多view切换,Picker,DataPicter实现
用Tab Bar Controller处理IPhone多个view切换, 而且还附有创建空项目,picker和DataPicker的实现! 具体步骤: 1.创建一个空项目,选择User Interfa ...
- Redis在Windows下的安装和使用
NoSQL简介 介绍redis前,我想还是先认识下NoSQL,即not only sql, 是一种非关系型的数据存储,key/value键值对存储.现有Nosql DB 产品: Redis/Mongo ...