题意:有一个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. SPOJ375 Query on a tree(LCT边权)

    之前做了两道点权的LCT,这次做一下边权的LCT.上网找了一下资料,发现对于边权的LCT有这么两种处理方法,一种是每条边建一个点,于是边权就转成点权了.另外一种则是每个边权对应到点权上,也就是每个点对 ...

  2. HDU 1392 Surround the Trees (Graham求凸包周长)

    题目链接 题意 : 让你找出最小的凸包周长 . 思路 : 用Graham求出凸包,然后对每条边求长即可. Graham详解 #include <stdio.h> #include < ...

  3. POJ 1185 炮兵阵地 (状压DP)

    题目链接 题意 : 中文题不详述. 思路 :状压DP,1表示该位置放炮弹,0表示不放.dp[i][j][k],代表第 i 行的状态为k时第i-1行的状态为 j 时放置的最大炮弹数.只是注意判断的时候不 ...

  4. WCF分布式开发步步为赢(2)自定义托管宿主WCF解决方案开发配置过程详解

    上一节<WCF分布式框架基础概念>我们介绍了WCF服务的概念和通信框架模型,并给出了基于自定义托管服务的WCF程序的实现代码.考虑到WCF分布式开发项目中关于托管宿主服务配置和客户端添加引 ...

  5. [转载] select, poll和epoll的区别

    源地址:http://sheepxxyz.blog.163.com/blog/static/61116213201022003513530/ 随着2.6内核对epoll的完全支持,网络上很多的文章和示 ...

  6. Maven的配置文件pom.xml

    Maven的配置文件pom.xml 简介: 什么是POM? POM是项目对象模型(Project Object Model)的简称,它是Maven项目中的文件,使用XML表示,名称叫做pom.xml. ...

  7. 使用 Nginx 和 GeoIP 模块来处理不同国家的访问

    安装 Nginx因为要用到 http_geoip_module 模块,系统自带的 nginx 一般不带这个模块,所以要下载 nginx 源代码后自行编译: # wget http://nginx.or ...

  8. JLINK固件,JLINK驱动和JLINK硬件版本之间的关系,以及固件升级方法

    初学者容易在这几个问题上面犯迷糊,这里简单的说说.   1. JLINK硬件版本首先说JLINK的硬件版本有V7,V8和V9,相信这一点大家应该都没问题,那怎么看自己手头的JLINK是哪个硬件版本呢, ...

  9. android 检测工具: drozer

    1. 下载 https://www.mwrinfosecurity.com/products/drozer/community-edition/ 可以下载windows的也可以下载linux的 2. ...

  10. 将EXE作为资源,然后在释放到磁盘上并运行该exe程序(使用了FindResource,LoadResource,然后用CFile写成一个文件)

    // 将exe作为资源加入,然后再释放出来,并运行 try { HRSRC hRes = FindResource(NULL, MAKEINTRESOURCE(IDR_EXE1), _T(" ...