题意:有一个n*m的矩阵,每个格子中有一个数字,或为0,或为1。有个人要从(1,1)到达(n,m),要求所走过的格子中的数字按先后顺序串起来后,用二进制的判断大小方法,让这个数字最小。前缀0不需要输出!!

思路:主要考虑的是BFS解决。

  如果grid[1,1]=1,那么这个二进制的位数也就定下来了,是n+m-1,很好解决,每个格子就只能往下或者往右,否则长度一定超过n+m+1,必定不是最优。

  如果grid[1,1]=0,那么可能会出现绕了一个S型到终点的结果为0而已。所以不能用老办法,要先预处理一下。处理方式是,用BFS将所有grid[1,1]可达的0点标记出来,找出其中距离终点最近的那些0点(可能多个),如果他们的下和右边的点为1,这些点都进队,再用上边方式BFS即可求得答案(上面只是1个起点,这边有多个起点,不影响正确性)。

  答案在哪?其实在BFS时每一层只能是0点或者是1点,为什么呢?如果有0点的话,还需要选择1点的吗?别忘了二进制的位数是固定了,选0肯定比选1要好,则在没有0的情况下再选1的。 在遍历时按层遍历,遍历到的点先分到两个集合A0和B1中,择所需即可,所以在遍历第i层时第i位的答案也就决定了。这是剪枝!

  注意考虑只有1个点,2个点和S形等各种情况。

 #include <bits/stdc++.h>
#define INF 0x7f7f7f7f
#define pii pair<int,int>
#define LL unsigned long long
using namespace std;
const int N=;
int n, m;
char grid[N][N];
vector<int> ans;
int inq[N][N]; void BFS(deque<pii> &que)
{
ans.push_back();
while(!que.empty())
{
deque<pii> que0, que1; //两种到达的方式,只取其一
int siz=que.size();
for(int i=; i<siz; i++) //被更新的都是同一源头的。
{
int a=que.front().first;
int b=que.front().second;
que.pop_front(); if( a+<=n && !inq[a+][b] ) //下:要么你无路径可达,要么我比你小,我才更新你
{
if(grid[a+][b]=='') que0.push_back(make_pair(a+,b));
else que1.push_back(make_pair(a+,b));
}
if( b+<=m && !inq[a][b+] ) //右
{
if(grid[a][b+]=='') que0.push_back(make_pair(a,b+));
else que1.push_back(make_pair(a,b+));
}
inq[a+][b]=inq[a][b+]=;
} if(!que0.empty()) ans.push_back();
else ans.push_back(); if(!que0.empty()) que.insert(que.end(), que0.begin(), que0.end() );
else que.insert(que.end(), que1.begin(), que1.end() );
}
} int cal()
{
memset(inq, , sizeof(inq)); deque<pii> que;que.push_back( make_pair(,));
if(grid[][]=='') //若起点为0,找到所有离终点最近的前缀0
{
inq[][]=;
while(!que.empty())
{
int siz=que.size();
for(int i=; i<siz; i++) //按层来BFS,按层记录最优答案
{
int a=que.front().first;
int b=que.front().second;
que.pop_front(); if(a+<=n && !inq[a+][b] && grid[a+][b]=='') que.push_back(make_pair(a+,b));
if(a-> && !inq[a-][b] && grid[a-][b]=='') que.push_back(make_pair(a-,b)); if(b+<=m && !inq[a][b+] && grid[a][b+]=='') que.push_back(make_pair(a,b+));
if(b-> && !inq[a][b-] && grid[a][b-]=='') que.push_back(make_pair(a,b-)); inq[a+][b]=inq[a-][b]=inq[a][b+]=inq[a][b-]=; //防止重复进队
}
}
int min_dis=INF;
for(int i=; i<=n; i++) //求最近的0距离终点的最小距离
{
for(int j=; j<=m; j++)
{
if(inq[i][j]&&grid[i][j]=='')
min_dis=min(min_dis, n+m-j-i);
}
}
if(grid[n][m]=='' && min_dis==) return ; //有0路可达终点
for(int i=; i<=n; i++) //扫出距离为min_dis的所有0点
for(int j=; j<=m; j++)
if(inq[i][j] && grid[i][j]=='' && min_dis==n+m-j-i && n+m-i-j!= )
que.push_back(make_pair(i,j)); memset(inq,,sizeof(inq));
int siz=que.size();
for(int i=; i<siz; i++) //将所有0点的下和右为1的点进队
{
int a=que.front().first;
int b=que.front().second;
que.pop_front();
if(a+<=n&&!inq[a+][b]&&grid[a+][b]=='') que.push_back(make_pair(a+,b));
if(b+<=m&&!inq[a][b+]&&grid[a][b+]=='') que.push_back(make_pair(a,b+));
inq[a+][b]=inq[a][b+]=;
}
}
BFS(que);
return ans.size();
} int main()
{
freopen("input.txt", "r", stdin);
int t, a, b;
char c;
cin>>t;
while(t--)
{
ans.clear();
scanf("%d %d",&n,&m); for(int i=; i<=n; i++) //输入要注意
for(int j=; j<=m; j++)
{
c=getchar();
if(c==''||c=='' ) grid[i][j]=c;
else j--;
}
int s=cal();
if(s==) printf("");
else for(int i=; i+<ans.size(); i++) printf("%d",ans[i]);//最后一个数字多余
printf("\n");
}
return ;
}

AC代码

HDU 5335 Walk Out (BFS,技巧)的更多相关文章

  1. HDU 5335 Walk Out BFS 比较坑

    H - H Time Limit:1000MS     Memory Limit:65536KB     64bit IO Format:%I64d & %I64u Submit Status ...

  2. hdu 5335 Walk Out(bfs+斜行递推) 2015 Multi-University Training Contest 4

    题意—— 一个n*m的地图,从左上角走到右下角. 这个地图是一个01串,要求我们行走的路径形成的01串最小. 注意,串中最左端的0全部可以忽略,除非是一个0串,此时输出0. 例: 3 3 001 11 ...

  3. hdu 5335 Walk Out(bfs+寻找路径)

    Problem Description In an n∗m maze, the right-bottom corner or a written on it. An explorer gets los ...

  4. hdu 5335 Walk Out (搜索)

    题目链接: hdu 5335 Walk Out 题目描述: 有一个n*m由0 or 1组成的矩形,探险家要从(1,1)走到(n, m),可以向上下左右四个方向走,但是探险家就是不走寻常路,他想让他所走 ...

  5. HDU 5335——Walk Out——————【贪心】

    Walk Out Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Su ...

  6. HDU 5335 Walk Out(多校)

    Walk Out Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Su ...

  7. hdu 5335 Walk Out (2015 Multi-University Training Contest 4)

    Walk Out                                                                         Time Limit: 2000/10 ...

  8. hdu 5335 Walk Out 搜索+贪心

    Walk Out Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total S ...

  9. 2015 Multi-University Training Contest 4 hdu 5335 Walk Out

    Walk Out Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Su ...

随机推荐

  1. Javascript 正则表达式笔记2

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  2. java 驼峰命名

    jstl中前台jsp页面调用对象中的属性时, 用的是小驼峰命名法. 例如:${item.createTime} 1.小驼峰式命名法(lower camel case): 第一个单字以小写字母开始,第二 ...

  3. sql server 2008 修改sa密码

    问题: 当我们用windows本身验证之后需要修改sa密码,出现这样的错误. 解决方案:

  4. IOS快速集成下拉上拉刷新

    http://code4app.com/ios/%E5%BF%AB%E9%80%9F%E9%9B%86%E6%88%90%E4%B8%8B%E6%8B%89%E4%B8%8A%E6%8B%89%E5% ...

  5. FastJson与Gson小测试

    最近用到Json来传输数据,找到两个比较简单的工具 Gson 和 FastJson随便测试一下两个工具的效率~ 1 package com.json.fast; import java.util.Ar ...

  6. JavaScript中函数的形参和实参的实现原理剖析

    我们都知道JS里面参数的传递是可以不一样的,比如我们有一个函数: <script type="text/javascript"> function one(a,b,c) ...

  7. iOS 安装使用cocoapods

    一.什么是CocoaPods 1.为什么需要CocoaPods 在进行iOS开发的时候,总免不了使用第三方的开源库,比如SBJson.AFNetworking.Reachability等等.使用这些库 ...

  8. iOS 苹果真机鉴定

    iPhone 4S(GSM) 16GB 黑色序列号:DX4KN69EDTC0设备名称:iPhone 4S容 量:16GB颜 色:黑色类 型:iPhone4,1代 号:n94ap型 号:MD235激活状 ...

  9. 转:Maven介绍(创建工程项目以及下载所需要的jar包)

    http://blog.csdn.net/guanghua2009/article/details/6644111 目前我还不太会用maven,慢慢学吧 目前就是用来下载自己要的jar包,先根据这篇转 ...

  10. The absolute uri: http://struts.apache.org/tags-bean cannot be resolved in either web.xml or the jar files deployed with this application

    在一个tomcat中部署了一个struts-1.3.10的web项目,但是没有吧struts-1.3.10的lib中的jar包放进tomcat/lib中,所以导致了这个错误(访问该项目的页面时)