传送门

题意

给定一个n*m的矩阵,询问q次,两个方块是否能被消掉,弯折次数不超过两次

分析

这题写了有一个下午,思路很简单,但是有很多trick,(唉),我还是太弱

trick

初始判断:1.两点不重叠

2.两点数值相等并且不为空

dfs中判断:1.每次访问节点深搜时打访问标记,回溯取消标记

2.一个强力剪枝

if(cnt==2&&(x-x2)&&(y-y2)) return ;

讲解:如果当到达一点弯折度已达2,并且该点与终点不在同一行/列,则返回。

强力剪枝!将我交的第一发8517ms降到124ms,很强!

代码

#include<cstdio>
#include<cstring>
#define R(i,a,b) for(int i=a;i<b;++i)
#define F(i,a,b) for(int i=a;i<=b;++i)
#define mem(a,b) memset(a,b,sizeof(a))
int t,n,m,a[1010][1010],q,x1,y1,x2,y2;
int dir[4][2]={0,1,1,0,0,-1,-1,0};
int flag;
void dfs(int x,int y,int cnt,int pre)
{
// printf("1.x=%d y=%d cnt=%d pre=%d\n",x,y,cnt,pre);
if(flag) return ;
if(x<1||y<1||x>n||y>m) return ;
if(cnt>2) return ;
if(cnt==2&&(x-x2)&&(y-y2)) return ;
//printf("(cnt==2&&(x-x2)&&(y-y2))=%d\n",(cnt==2&&(x-x2)&&(y-y2)));
//if(cnt==2&&(x-x2)&&(y-y2)) return ;
if(x2==x&&y2==y) { flag=1;return ; }
//printf("a[%d][%d]=%d\n",x,y,a[x][y]);
//printf("2.x=%d y=%d cnt=%d pre=%d\n",x,y,cnt,pre);
R(i,0,4)
{
int xx=x+dir[i][0],yy=y+dir[i][1];
int cnt1;
if(pre!=i) cnt1=cnt+1;else cnt1=cnt;
if(a[xx][yy]==0)
{
a[xx][yy]=1;
//printf("3.a[%d][%d]=%d\n",xx,yy,a[xx][yy]);
dfs(xx,yy,cnt1,i);
a[xx][yy]=0;
}
else if(xx==x2&&yy==y2)
{
dfs(xx,yy,cnt1,i);
if(flag) return ;
}
// printf("3.x=%d y=%d cnt=%d pre=%d\n",xx,yy,cnt,pre);
//if(pre!=i) dfs(xx,yy,cnt+1,i);else dfs(xx,yy,cnt,i);
}
}
int main()
{
while(scanf("%d %d",&n,&m),n+m)
{
F(i,1,n)F(j,1,m) scanf("%d",&a[i][j]);
for(scanf("%d",&q);q--;)
{
scanf("%d %d %d %d",&x1,&y1,&x2,&y2);
if(a[x1][y1]!=a[x2][y2]||a[x1][y1]==0||(x1==x2&&y1==y2)) { flag=0;goto l; }
flag=0;
R(i,0,4)
{
int xx=x1+dir[i][0],yy=y1+dir[i][1];
if(xx<1||yy<1||xx>n||yy>m) continue;
if(flag) break;
if(a[xx][yy]==0)
{
a[xx][yy]=1;//已访问过
dfs(xx,yy,0,i);
a[xx][yy]=0;//回溯清标记
}
else if(xx==x2&&yy==y2)
{
dfs(xx,yy,0,i);
}
}
l:if(flag) puts("YES");else puts("NO");
}
}
}

HDU1175:连连看(搜索)的更多相关文章

  1. hdu1175 连连看(bfs疯狂MLE和T,遂考虑dfs+剪枝)

    题目链接:http://icpc.njust.edu.cn/Problem/Hdu/1175/ 题目大意就是给出地图,上面有若干的数,相当于连连看,给了q个查询,问给出的两个位置的数能否在两次转弯以内 ...

  2. hdu1175连连看

    Problem Description “连连看”相信很多人都玩过.没玩过也没关系,下面我给大家介绍一下游戏规则:在一个棋盘中,放了很多的棋子.如果某两个相同的棋子,可以通过一条线连起来(这条线不能经 ...

  3. HDU1175 连连看(DFS)

    Problem Description “连连看”相信很多人都玩过.没玩过也没关系,下面我给大家介绍一下游戏规则:在一个棋盘中,放了很多的棋子.如果某两个相同的棋子,可以通过一条线连起来(这条线不能经 ...

  4. HDU1175 连连看(bfs) 2016-07-24 13:27 115人阅读 评论(0) 收藏

    连连看 Problem Description "连连看"相信很多人都玩过.没玩过也没关系,下面我给大家介绍一下游戏规则:在一个棋盘中,放了很多的棋子.如果某两个相同的棋子,可以通 ...

  5. hdu1175连连看(dfs+细节)

    连连看 Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submi ...

  6. hdu1175 连连看

    连连看 HDU - 1175 “连连看”相信很多人都玩过.没玩过也没关系,下面我给大家介绍一下游戏规则:在一个棋盘中,放了很多的棋子.如果某两个相同的棋子,可以通过一条线连起来(这条线不能经过其它棋子 ...

  7. 连连看[HDU1175]

    连连看 Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submi ...

  8. 连连看(简单搜索)bfs

    连连看Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submis ...

  9. 连连看 HDU - 1175_搜索_剪枝

    hdu有毒,考试上 AC 的就是一直 WA- 其实这道题是可以进行初始化来进行优化的,这样的话询问次数是可以达到 10510^5105 的.不过普通的 dfsdfsdfs + 剪枝也是可过的. Cod ...

随机推荐

  1. THUPC2018看题总结

    THUPC2018看题总结 #6387. 「THUPC2018」绿绿与串串 / String 据说是签到题啊. 首先根据题目的意思,我们发现如果能找到那个最后一次选择的对称轴岂不是美滋滋. 自然地,我 ...

  2. Date日期模式

    package cn.zmh.Date; import java.text.SimpleDateFormat; import java.util.Date; public class DateDemo ...

  3. Java: 创建自带依赖库的Jar包

    pom.xml文件如下: <?xml version="1.0" encoding="UTF-8"?> <project xmlns=&quo ...

  4. Python遍历路径下文件并转换成UTF-8编码

    http://www.cnblogs.com/wuyuegb2312/archive/2013/01/11/2856772.html 开始学Python,这篇文章来自于应用需求. os.walk很方便 ...

  5. 学习Centos 7的笔记

    Step-1 yum install epel-release && yum clean all && yum update –y &&  yum -y ...

  6. 解决java.util.concurrent.ExecutionException: org.apache.catalina.LifecycleException

    解决: 工程目录.settings\org.eclipse.wst.common.project.facet.core.xml文件中jst.web的version降低到2.5 <?xml ver ...

  7. centos中w使用smbclient连接window出现:session setup failed: NT_STATUS_LOGON_FAILURE

    1. 在window中网络->我自己的电脑->能够查看到共享文件,说明window的共享是正常了; 2. 在window中配置共享时,使用的是仅仅同意超级管理员訪问,可是我把超级管理员改名 ...

  8. SQL 连接(JOIN)

    SQL 连接(JOIN) SQL join 用于把来自两个或多个表的行结合起来. SQL JOIN SQL JOIN 子句用于把来自两个或多个表的行结合起来,基于这些表之间的共同字段. 最常见的 JO ...

  9. CodeForces 404 Marathon ( 浮点数取模 -- 模拟 )

    B. Marathon time limit per test 1 second memory limit per test 256 megabytes input standard input ou ...

  10. 设计模式C++实现_2_简单工厂模式

    简单工厂模式 主要用于创建对象. 新加入类时. 不会影响曾经的系统代码. 核心思想是用一个工厂来依据输入的条件产生不同的类,然后依据不同类的 virtual 函数得到不同的结果. 以下以苹果手机的生产 ...