poj3009 Curling 2.0

Posted patrolli

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了poj3009 Curling 2.0相关的知识,希望对你有一定的参考价值。

题目

一场冰面比赛在一个矩形区域进行,区域上设置有障碍(1表示),空白区域(0表示),起点(2表示),终点(3表示)。起始在起点有一个石头,每次将它向上左下右四个方向之一抛出(当某一方向上临近有障碍时不能向这一方向抛出),石块将沿这一方向运动,直到:

  • 石头撞到了障碍:石头停在靠近障碍的那个空格上,障碍消失
  • 石头超出了矩形区域的范围,这次抛出无效
  • 到达终点

一局游戏一共可以抛掷石头10次,如果10次后石头还没有到达终点,那么游戏失败。需要到达终点的最少抛掷次数。

样例输入

2 1
3 2
6 6
1 0 0 2 1 0
1 1 0 0 0 0
0 0 0 0 0 3
0 0 0 0 0 0
1 0 0 0 0 1
0 1 1 1 1 1
6 1
1 1 2 1 1 3
6 1
1 0 2 1 1 3
12 1
2 0 1 1 1 1 1 1 1 1 1 3
13 1
2 0 1 1 1 1 1 1 1 1 1 1 3
0 0

样例输出

1
4
-1
4
10
-1

思路

  • _throw()函数模拟每次石头向某一方向运动,返回三种结果:出界、到达终点、遇到障碍。
  • 根据抛出的三种结果,出界则转向下一方向尝试,到达终点则更新最终步数,遇到障碍则继续递归到新一状态,从这一状态开始重新将石头向四个方向扔出。注意保留现场。

代码

#include <iostream>
#include <utility>
using namespace std;
typedef pair<int, int> PII;
const int N = 22;
const int dr[] = {0, 1, -1, 0};
const int dc[] = {1, 0, 0, -1};
int m, n;
int end_r, end_c; //终点的坐标
int g[N][N];
int cnt, steps;

int _throw(int start_r, int start_c, int dir, PII &end_point){
    int raw = start_r, col = start_c;
    while(1){
        raw += dr[dir];
        col += dc[dir];
    //   printf("cur_r: %d cur_c: %d
", raw, col);
        if(raw < 0 || raw >= m || col < 0 || col >= n){ // outside
            end_point.first = start_r;
            end_point.second = start_c;
            return 0;
        }
        if(raw == end_r && col == end_c){  //arrive the end point
            return 1;
        }
        if(g[raw][col] == 1){     //arrive a block
            end_point.first = raw - dr[dir];
            end_point.second = col - dc[dir];
            return 2;
        }
    }
}

void dfs(int start_x, int start_y){
    if(cnt > 10) return;
    int condition;
    for(int dir = 0; dir < 4; ++dir){
        PII end_point;
        end_point.first = start_x; end_point.second = start_y;
        int tmpr = start_x + dr[dir];
        int tmpc = start_y + dc[dir];                //用于判断“起点”四周是否有障碍阻挡抛出
        if(tmpr>=0 && tmpr<m && tmpc>=0 && tmpc < n && g[tmpr][tmpc] != 1){ //当四周没有blocks时可以throw
            condition = _throw(start_x, start_y, dir, end_point);
            if(condition == 0) continue;
            if(condition == 1){ //到达终点
                if(cnt+1 < steps) {steps  = cnt+1;} //更新最少到达的结果到steps中
                continue;
            }
            if(condition == 2) {    
                int old_cnt = cnt;  //保留当前状态的步数
                cnt++;
                g[end_point.first+dr[dir]][end_point.second+dc[dir]] = 0; //清除障碍
                dfs(end_point.first, end_point.second); //进入下一状态
                g[end_point.first+dr[dir]][end_point.second+dc[dir]] = 1; //恢复现场
                cnt = old_cnt;
            }
        }
    }
}

int main(){
    //freopen("input.txt", "r", stdin);
    //freopen("output.txt", "w", stdout);
    int start_r, start_c;
    while(scanf("%d%d", &n, &m) && m != 0){
        for(int i = 0; i < m; ++i){
            for(int j = 0; j < n; ++j){
                scanf("%d", &g[i][j]);
                if(g[i][j] == 2){ start_r = i; start_c = j;}
                if(g[i][j] == 3) { end_r = i; end_c = j;}
            }
        } 
        
        cnt = 0, steps = 100000;
        dfs(start_r, start_c);
        if(steps>10) printf("-1
");
        else printf("%d
", steps);
    }
}

以上是关于poj3009 Curling 2.0的主要内容,如果未能解决你的问题,请参考以下文章

Curling 2.0 POJ 3009(搜索)

poj 3009 Curling 2.0

poj3009 Curling 2.0(很好的题 DFS)

POJ 3009 Curling 2.0

poj 3009 Curling 2.0

POJ 3009 ( Curling 2.0 )