HDU 1813 Escape from Tetris IDA*

2024-04-22 07:38
文章标签 ida hdu escape tetris 1813

本文主要是介绍HDU 1813 Escape from Tetris IDA*,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题意:
由于整日整夜地对着这个棋盘,Lele终于走火入魔。每天一睡觉,他就会梦到自己会被人被扔进一个棋盘中,一直找不到出路,然后从梦中惊醒。久而久之,Lele被搞得精神衰弱。梦境是否会成为现实,谁也说不准,不过不怕一万只怕万一。现在Lele每次看到一个棋盘,都会想象一下自己被关进去以后要如何逃生。

Lele碰到的棋盘都是正方形的,其中有些格子是坏的,不可以走,剩下的都是可以走的。只要一走到棋盘的边沿(最外面的一圈),就算已经逃脱了。Lele梦见自己一定会被扔在一个可以走的格子里,但是不确定具体是哪一个,所以他要做好被扔在任意一个格子的准备。

现在Lele请你帮忙,对于任意一个棋盘,找出一个最短的序列,序列里可以包括"north"(地图里向上),"east"(地图里向右),"south"(地图里向下),"west"(地图里向左),这四个方向命令。不论Lele被扔在棋盘里的哪个好的格子里,都能按这个序列行走逃出棋盘。
逃脱的具体方法是:不论Lele被扔在哪里,Lele按照序列里的方向命令一个一个地走,每个命令走一格,如果走的时候会碰到坏的格子,则忽略这条命令。当然,如果已经逃脱了,就可以不考虑序列中剩下的命令了。如果存在两个符合要求的序列,请输出字典序最小的那个序列。

想法:有的人可能不需要看题解,但就是wa,先上一个我没注意的坑点,字典序最小,这个条件,我做题目做做就忘了这个条件了,一直wa,由于要字典序最小,所以枚举方向的时候
一定要是east,north,south,west,因为字母的顺序就是ensw,尴尬了,我就没注意永远上北下南左西右东。wa了十几发还不知道错误。
说完这个坑点,就说说思路了:首先我们先找到,每一个点到边界的最短距离,这为之后用到,那么定义dis[i][j],表示(i,j)到边界的最短距离。然后就枚举方向,然后图中每一个可行点都要
朝着这个方向走。如果下一步不通,那么此点原地不动,如果下一步可行,则保存行动之后的点。如果所有点到边界的距离都为0那么,表示存在解,输出即可。
这里有一个减枝if:d+get_h(mat)>step then:return false, get_h()函数返回的是所有点中到达边界最短距离中的最大值,如果加上已有的走的步数大于设定的步数,那就不需要再继续了,直接返回false即可。
 

#include<iostream>
#include<cstring>
#include<queue>
#define inf 0x7fffffff 
using namespace std;
char map[9][9];
char Dir[4][10] = {"east","north","south","west"}; 
int dir[4][2] = {0,1,-1,0,1,0,0,-1}; 
int dis[9][9],path[1000];
int n,step;
struct node
{int x,y;
};
queue<node>q;
bool isedge(int x,int y)
{return x==0||x==n-1||y==0||y==n-1;
}
bool ismap(int i,int j)
{return i>=0&&i<n&&j>=0&&j<n;
}
void bfs()
{node head,tail;while(!q.empty()){head=q.front();q.pop();for(int i=0;i<4;i++){tail.x=head.x+dir[i][0];tail.y=head.y+dir[i][1];if(!map[tail.x][tail.y]) continue;if(!ismap(tail.x,tail.y)) continue;if(dis[tail.x][tail.y]>dis[head.x][head.y]+1){dis[tail.x][tail.y]=dis[head.x][head.y]+1;q.push(tail);}}}
} 
void Init()
{while(!q.empty()) q.pop();for(int i=0;i<n;i++){for(int j=0;j<n;j++){dis[i][j]=inf;map[i][j]=map[i][j]=='1'?0:1;if(!map[i][j]) continue;if(isedge(i,j)){dis[i][j]=0;node pp;pp.x=i;pp.y=j;q.push(pp);}}}
} 
int get_h(char mat[9][9])
{int Mx=0;for(int i=0;i<n;i++){for(int j=0;j<n;j++){if(mat[i][j]) Mx=max(Mx,dis[i][j]);}}return Mx;
} 
bool dfs(char mat[9][9],int d)
{if(d+get_h(mat)>step) return false;if(d==step) return true;char nxt[9][9];for(int k=0;k<4;k++){memset(nxt,0,sizeof(nxt));for(int i=0;i<n;i++){for(int j=0;j<n;j++){if(isedge(i,j)||!mat[i][j]) continue;int nx=i+dir[k][0];int ny=j+dir[k][1];if(!map[nx][ny]) nxt[i][j]=1;else nxt[nx][ny]=1;}}path[d]=k;if(dfs(nxt,d+1)) return true;}return false;
}
int main()
{int ca=0;while(~scanf("%d",&n)){if(ca++) cout<<endl;for(int i=0;i<n;i++)scanf("%s",map[i]);Init();bfs();step=0;while(1){if(dfs(map,0)) break;step++;}for(int i=0;i<step;i++){printf("%s\n",Dir[path[i]]);}}return 0;
}


这篇关于HDU 1813 Escape from Tetris IDA*的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/925173

相关文章

usaco 1.3 Mixing Milk (结构体排序 qsort) and hdu 2020(sort)

到了这题学会了结构体排序 于是回去修改了 1.2 milking cows 的算法~ 结构体排序核心: 1.结构体定义 struct Milk{int price;int milks;}milk[5000]; 2.自定义的比较函数,若返回值为正,qsort 函数判定a>b ;为负,a<b;为0,a==b; int milkcmp(const void *va,c

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

hdu 2093 考试排名(sscanf)

模拟题。 直接从教程里拉解析。 因为表格里的数据格式不统一。有时候有"()",有时候又没有。而它也不会给我们提示。 这种情况下,就只能它它们统一看作字符串来处理了。现在就请出我们的主角sscanf()! sscanf 语法: #include int sscanf( const char *buffer, const char *format, ... ); 函数sscanf()和

hdu 2602 and poj 3624(01背包)

01背包的模板题。 hdu2602代码: #include<stdio.h>#include<string.h>const int MaxN = 1001;int max(int a, int b){return a > b ? a : b;}int w[MaxN];int v[MaxN];int dp[MaxN];int main(){int T;int N, V;s

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

hdu 1166 敌兵布阵(树状数组 or 线段树)

题意是求一个线段的和,在线段上可以进行加减的修改。 树状数组的模板题。 代码: #include <stdio.h>#include <string.h>const int maxn = 50000 + 1;int c[maxn];int n;int lowbit(int x){return x & -x;}void add(int x, int num){while

hdu 3790 (单源最短路dijkstra)

题意: 每条边都有长度d 和花费p,给你起点s 终点t,要求输出起点到终点的最短距离及其花费,如果最短距离有多条路线,则输出花费最少的。 解析: 考察对dijkstra的理解。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstrin

hdu 2489 (dfs枚举 + prim)

题意: 对于一棵顶点和边都有权值的树,使用下面的等式来计算Ratio 给定一个n 个顶点的完全图及它所有顶点和边的权值,找到一个该图含有m 个顶点的子图,并且让这个子图的Ratio 值在所有m 个顶点的树中最小。 解析: 因为数据量不大,先用dfs枚举搭配出m个子节点,算出点和,然后套个prim算出边和,每次比较大小即可。 dfs没有写好,A的老泪纵横。 错在把index在d

hdu 1102 uva 10397(最小生成树prim)

hdu 1102: 题意: 给一个邻接矩阵,给一些村庄间已经修的路,问最小生成树。 解析: 把已经修的路的权值改为0,套个prim()。 注意prim 最外层循坏为n-1。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstri

hdu 1285(拓扑排序)

题意: 给各个队间的胜负关系,让排名次,名词相同按从小到大排。 解析: 拓扑排序是应用于有向无回路图(Direct Acyclic Graph,简称DAG)上的一种排序方式,对一个有向无回路图进行拓扑排序后,所有的顶点形成一个序列,对所有边(u,v),满足u 在v 的前面。该序列说明了顶点表示的事件或状态发生的整体顺序。比较经典的是在工程活动上,某些工程完成后,另一些工程才能继续,此时