zoukankan      html  css  js  c++  java
  • HDU 1175 连连看(BFS或DFS)

    连连看

                  Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)

                           Total Submission(s): 12814    Accepted Submission(s): 3351

    Problem Description
    “连连看”相信很多人都玩过。没玩过也没关系,下面我给大家介绍一下游戏规则:在一个棋盘中,放了很多的棋子。如果某两个相同的棋子,可以通过一条线连起来(这条线不能经过其它棋子),而且线的转折次数不超过两次,那么这两个棋子就可以在棋盘上消去。不好意思,由于我以前没有玩过连连看,咨询了同学的意见,连线不能从外面绕过去的,但事实上这是错的。现在已经酿成大祸,就只能将错就错了,连线不能从外围绕过。 玩家鼠标先后点击两块棋子,试图将他们消去,然后游戏的后台判断这两个方格能不能消去。现在你的任务就是写这个后台程序。
     
    Input
    输入数据有多组。每组数据的第一行有两个正整数n,m(0<n<=1000,0<m<1000),分别表示棋盘的行数与列数。在接下来的n行中,每行有m个非负整数描述棋盘的方格分布。0表示这个位置没有棋子,正整数表示棋子的类型。接下来的一行是一个正整数q(0<q<50),表示下面有q次询问。在接下来的q行里,每行有四个正整数x1,y1,x2,y2,表示询问第x1行y1列的棋子与第x2行y2列的棋子能不能消去。n=0,m=0时,输入结束。 注意:询问之间无先后关系,都是针对当前状态的!
     
    Output
    每一组输入数据对应一行输出。如果能消去则输出"YES",不能则输出"NO"。
     
    Sample Input
    3 4
    1 2 3 4
    0 0 0 0
    4 3 2 1
    4
    1 1 3 4
    1 1 2 4
    1 1 3 3
    2 1 2 4
    3 4
    0 1 4 3
    0 2 4 1
    0 0 0 0
    2
    1 1 2 4
    1 3 2 3
    0 0
     
    Sample Output
    YES
    NO
    NO
    NO
    NO
    YES
     
     
     
    方法一:用BFS求解
      1 #include <stdio.h>
      2 #include <string.h>
      3 
      4 typedef struct _QUEUE           //定义一个队列,保存相关信息
      5 {
      6     int preX, preY;
      7     int curX, curY;
      8     int turnCount;
      9 }QUEUE;
     10 
     11 int M, N;                       //N代表行,M代表列
     12 int sRow, sCol;                 //起始点
     13 int eRow, eCol;                 //到达的终点
     14 const int MAX_NUM = 1005;
     15 int   maze[MAX_NUM][MAX_NUM];   //记录迷宫
     16 int   step[MAX_NUM][MAX_NUM];   //记录某个点的最小拐弯次数
     17 QUEUE qu[MAX_NUM * MAX_NUM];    //对每个符合条件的点入队列
     18 int   dir[4][2] = {{0,-1}, {0,1}, {-1,0}, {1,0}};
     19 
     20 bool bfs()
     21 {
     22     memset(step, 0x7f, sizeof(step)); //将每个点的最小拐弯次数初始为正无穷
     23     qu[0].preX = qu[0].curX = sRow;   //起点入队列
     24     qu[0].preY = qu[0].curY = sCol;
     25     qu[0].turnCount = 0;
     26 
     27     int front = -1, tail = 0;
     28     while(front < tail)              //队列不空时循环
     29     {
     30         QUEUE tempP = qu[++front];
     31         if(tempP.curX == eRow && tempP.curY == eCol) //找到出口
     32             return true;
     33 
     34         for(int k = 0; k < 4; k++)   //向四个方向拓展
     35         {
     36             QUEUE tempC;
     37             tempC.preX = tempP.curX; //计算下一个点的相关信息
     38             tempC.preY = tempP.curY;
     39             tempC.curX = tempP.curX + dir[k][0];
     40             tempC.curY = tempP.curY + dir[k][1];
     41 
     42             if(tempC.curX > 0 && tempC.curX <= N && tempC.curY > 0 
     43                 && tempC.curY <= M && maze[tempC.curX][tempC.curY] == 0)    
     44             {
     45                 if( (tempP.preX == tempP.curX && tempP.curX == tempC.curX)
     46                     || (tempP.preY == tempP.curY && tempP.curY == tempC.curY) )
     47                 {
     48                     tempC.turnCount = tempP.turnCount;    //在同一条直线
     49                 }
     50                 else
     51                 {
     52                     tempC.turnCount = tempP.turnCount + 1;//不在同一条直线
     53                 }
     54 
     55                    //若拐点小于等于2,且小于step[][],主要是防止将已经走过的点入队列
     56                 if(tempC.turnCount <= 2 && tempC.turnCount < step[tempC.curX][tempC.curY])
     57                 {
     58                     step[tempC.curX][tempC.curY] = tempC.turnCount;
     59                     qu[++tail] = tempC;                  //符合条件的点入队列
     60                 }
     61             }
     62         }
     63     }
     64     return false; //找不到出口,返回false
     65 }
     66 
     67 int main()
     68 {
     69     while(scanf("%d %d", &N, &M) && M+N)
     70     {
     71         for(int row = 1; row <= N; row++)
     72         {
     73             for(int col = 1; col <= M; col++)
     74             {
     75                 scanf("%d", &maze[row][col]);
     76             }
     77         }
     78 
     79         int Q;
     80         scanf("%d", &Q);
     81         while(Q--)
     82         {
     83             scanf("%d %d %d %d", &sRow, &sCol, &eRow, &eCol);
     84 
     85             if(maze[sRow][sCol] == 0 || maze[sRow][sCol] != maze[eRow][eCol]
     86                 || maze[eRow][eCol] == 0 || (sRow == eRow && sCol == eCol))
     87             {
     88                 printf("NO\n");
     89                 continue;
     90             }
     91 
     92             int tempValue = maze[eRow][eCol];//将终点值赋给临时变量
     93             maze[eRow][eCol] = 0;            //将终点位置置为0
     94 
     95             bool bResult = bfs();            //BFS搜索
     96 
     97             if(bResult)
     98                 printf("YES\n");
     99             else
    100                 printf("NO\n");
    101 
    102             maze[eRow][eCol] = tempValue;   //恢复终点位置的值
    103         }
    104     }
    105     return 0;
    106 }

    方法二:DFS求解

     1 #include <stdio.h>
     2 #include <string.h>
     3 
     4 int M, N;                           //N代表行,M代表列
     5 int sRow, sCol;                     //起始点
     6 int eRow, eCol;                     //到达的终点
     7 const int MAX_NUM = 1005;
     8 int   maze[MAX_NUM][MAX_NUM];       //记录迷宫
     9 int   visit[MAX_NUM][MAX_NUM];      //记录某个点是否被访问
    10 int   dir[4][2] = {{0,-1}, {0,1}, {-1,0}, {1,0}};
    11 bool  bResult;                      //记录结果
    12 
    13 //分别记录前一个点位置,当前点的位置,以及当前点的拐弯次数
    14 void dfs(int preX, int preY, int curX, int curY, int turnCount)
    15 {
    16     if(bResult)return;               //已经找到方案则返回
    17     if(turnCount > 2)return;         //当前点的拐弯次数大于2,直接返回
    18 
    19     if(curX == eRow && curY == eCol) //找到终点,置bResult为true
    20     {
    21         bResult = true;
    22         return ;
    23     }
    24 
    25     for(int k = 0; k < 4; k++)       //向四个方向拓展
    26     {
    27         int nextX = curX + dir[k][0];
    28         int nextY = curY + dir[k][1];
    29 
    30         if(nextX > 0 && nextX <= N && nextY > 0 && nextY <= M) //在迷宫范围内
    31         {
    32             if(!visit[nextX][nextY] && maze[nextX][nextY] == 0)//没有访问且值为0
    33             {
    34                 int nTurn;
    35                 if( (preX==curX&&curX==nextX || preY==curY&&curY==nextY) )
    36                 {
    37                     nTurn = turnCount;       //若在一条线上,拐点数目不增加
    38                 }
    39                 else
    40                 { 
    41                     nTurn = turnCount + 1;   //不在同一条线上,拐点次数加一
    42                 }
    43                 
    44                 visit[nextX][nextY] = 1;     //标记为已访问
    45                 dfs(curX, curY, nextX, nextY, nTurn);  //搜索下一个点
    46                 visit[nextX][nextY] = 0;     //回溯时,恢复原来状态
    47             }
    48         }
    49     }
    50 }
    51 
    52 int main()
    53 {
    54     while(scanf("%d %d", &N, &M) && M+N)
    55     {
    56         for(int row = 1; row <= N; row++)
    57         {
    58             for(int col = 1; col <= M; col++)
    59             {
    60                 scanf("%d", &maze[row][col]);
    61             }
    62         }
    63 
    64         int Q;
    65         scanf("%d", &Q);
    66         while(Q--)
    67         {
    68             scanf("%d %d %d %d", &sRow, &sCol, &eRow, &eCol);
    69 
    70             if(maze[sRow][sCol] == 0 || maze[sRow][sCol] != maze[eRow][eCol] 
    71                 || maze[eRow][eCol] == 0 || (sRow == eRow && sCol == eCol))
    72             {
    73                 printf("NO\n");
    74                 continue;
    75             }
    76 
    77             bResult = false;
    78             memset(visit, 0, sizeof(visit));
    79 
    80             int tempValue = maze[eRow][eCol]; //将终点值赋给临时变量
    81             maze[eRow][eCol] = 0;             //将终点位置置为0
    82 
    83             dfs(sRow, sCol, sRow, sCol, 0);   //DFS搜索
    84 
    85             if(bResult)
    86                 printf("YES\n");
    87             else
    88                 printf("NO\n");
    89 
    90             maze[eRow][eCol] = tempValue;     //恢复终点位置的值
    91         }
    92     }
    93     return 0;
    94 }
  • 相关阅读:
    css列表
    css字体
    css文本
    css背景
    css里的属性
    MySQL语法大全_自己整理的学习笔记
    必看的 jQuery性能优化的38个建议
    p​h​p​异​常​机​制
    常用软件
    php过滤危险html代码
  • 原文地址:https://www.cnblogs.com/Dreamcaihao/p/3103304.html
Copyright © 2011-2022 走看看