码迷,mamicode.com
首页 > 其他好文 > 详细

poj 2110 Mountain Walking 枚举+bfs

时间:2015-05-21 17:30:36      阅读:144      评论:0      收藏:0      [点我收藏+]

标签:poj   算法   

题意:

给一个n*n的矩阵,要从左上角走到右下角,使经过数字的最大数与最小数的差最小。

分析:

一开始想到了二分这个差,然后判断是否存在路径,每次只知道差的话深搜每次搜索要记录沿途的最大值和最小值会tle,广搜的话如果节点只记录x,y坐标,搜索中存在要重新访问以前访问过节点的情况,比如一开始(1,1)->(1,2)->(2,2),如果(2,1)这个点的值更合适,最优访问路径(1,1)->(2,1)->(2,2),也就是(2,2)要被重新访问,不满足广搜每个节点只访问一次的原则,只有增加节点维度每个节点记录x,y坐标和到达它经过的最小最大值,显然不好。后来了解到可以加大枚举,不仅枚举差,还枚举路径上的最大值,这样每次路径上的最大最小值就确定了,可以广搜。

//poj 2110
//sep9
#include <iostream>
#include <queue>
using namespace std;
const int maxN=128;
struct Node
{
	int x,y;		
};

int g[maxN][maxN];
int vis[maxN][maxN];
int dirx[4]={0,0,-1,1};
int diry[4]={-1,1,0,0};
int n,diff,min_hight,max_hight;
queue<Node> q;


bool pass(int low,int high)
{
	memset(vis,0,sizeof(vis));
	Node a;
	a.x=1,a.y=1;
	if(g[1][1]>high||g[1][1]<low)
		return false;
	while(!q.empty()) q.pop();
	q.push(a);
	vis[1][1]=1;
	while(!q.empty()){
		Node t=q.front();q.pop();
		int x=t.x,y=t.y;	
		for(int k=0;k<4;++k){
			int nx=x+dirx[k];
			int ny=y+diry[k];
			if(nx>=1&&nx<=n&&ny>=1&&ny<=n&&vis[nx][ny]==0&&g[nx][ny]<=high&&g[nx][ny]>=low){
				Node a;
				a.x=nx,a.y=ny;
				q.push(a);
				vis[nx][ny]=1;
				if(nx==n&&ny==n) return true;
			}	
		}
	}
	return false;		
}

bool work(int mid)
{
	for(int high=min_hight;high<=max_hight;++high){
		int low=high-mid;	
		if(pass(low,high))
			return true;
	}
	return false;

}

int main()
{
	scanf("%d",&n);
	min_hight=INT_MAX;
	max_hight=INT_MIN; 
	for(int i=1;i<=n;++i)
		for(int j=1;j<=n;++j){
			scanf("%d",&g[i][j]);
			min_hight=min(min_hight,g[i][j]);
			max_hight=max(max_hight,g[i][j]); 
		}
	int ans,l=0,r=max_hight+1,mid;
	while(l<r){
		mid=(l+r)/2;	
		if(work(mid)){
			r=mid;
			ans=mid;
		}else
			l=mid+1;
	}
	printf("%d",ans);	
	return 0;	
} 


poj 2110 Mountain Walking 枚举+bfs

标签:poj   算法   

原文地址:http://blog.csdn.net/sepnine/article/details/45893151

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!