题意:从起点出发,可向东南西北4个方向走,如果前面没有墙则可走;如果前面只有一堵墙,则可将墙向前推一格,其余情况不可推动,且不能推动游戏区域边界上的墙。问走出迷宫的最少步数,输出任意一个移动序列。

分析:

1、最少步数--IDA*。

2、注意,若此墙可推动,必须改变当前格子,和沿当前格子向前一步的格子的墙的标记。

3、若沿当前格子向前两步的格子存在,则这个格子的墙的标记也要改变。不存在的情况是:把墙推向了边界。

4、因为每个格子的值是是1(如果正方形以西有一个墙),2(北),4(东)和8(南)的总和,所以通过与1,2,4,8 & 分别来判断西,北,东,南四个方向是否有墙。

5、可以通过当前步数与当前格子和最近边界的垂直距离的和是否大于maxn来剪枝。

6、注意起点要标记成已走过。

#pragma comment(linker, "/STACK:102400000, 102400000")
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<cctype>
#include<cmath>
#include<iostream>
#include<sstream>
#include<iterator>
#include<algorithm>
#include<string>
#include<vector>
#include<set>
#include<map>
#include<stack>
#include<deque>
#include<queue>
#include<list>
#define Min(a, b) ((a < b) ? a : b)
#define Max(a, b) ((a < b) ? b : a)
typedef long long ll;
typedef unsigned long long llu;
const int INT_INF = 0x3f3f3f3f;
const int INT_M_INF = 0x7f7f7f7f;
const ll LL_INF = 0x3f3f3f3f3f3f3f3f;
const ll LL_M_INF = 0x7f7f7f7f7f7f7f7f;
const int dr[] = {0, -1, 0, 1, -1, -1, 1, 1};//西北东南
const int dc[] = {-1, 0, 1, 0, -1, 1, -1, 1};
const int MOD = 1e9 + 7;
const double pi = acos(-1.0);
const double eps = 1e-15;
const int MAXN = 50 + 10;
const int MAXT = 10000 + 10;
using namespace std;
int pic[10][10];
int vis[10][10];
int ans[30];
int maxn;
const string s = "WNES";
int dir[] = {1, 2, 4, 8};
int solve(int x, int y){
if(x == 1 && !(pic[x][y] & 2)) return 1;//此刻位于迷宫最北面,且当前格子北面没有墙
if(x == 4 && !(pic[x][y] & 8)) return 3;//东
if(y == 1 && !(pic[x][y] & 1)) return 0;//北
if(y == 6 && !(pic[x][y] & 4)) return 2;//南
return -1;
}
bool judge(int x, int y){
return x >= 1 && x <= 4 && y >= 1 && y <= 6;
}
bool dfs(int x, int y, int cur){
if(cur >= maxn) return false;
int tmp = solve(x, y);
if(tmp != -1){
ans[cur] = tmp;
return true;
}
for(int i = 0; i < 4; ++i){
int tx = x + dr[i];
int ty = y + dc[i];
if(judge(tx, ty) && !vis[tx][ty]){
if(!(pic[x][y] & dir[i])){//向pic[tx][ty]方向走,无墙
vis[tx][ty] = 1;
ans[cur] = i;
if(dfs(tx, ty, cur + 1)) return true;
vis[tx][ty] = 0;
}
else if(!(pic[tx][ty] & dir[i])){//向pic[tx][ty]方向走,有墙但可推
vis[tx][ty] = 1;
pic[tx][ty] += dir[i];//墙推动导致格子对墙的标记改变
pic[x][y] -= dir[i];
if(judge(tx + dr[i], ty + dc[i])){
pic[tx + dr[i]][ty + dc[i]] += dir[(i + 2) % 4];//注意对于该格子加反方向的墙,但是该格子可能不存在,比如把墙推向了边界
}
ans[cur] = i;
if(dfs(tx, ty, cur + 1)) return true;
if(judge(tx + dr[i], ty + dc[i])){
pic[tx + dr[i]][ty + dc[i]] -= dir[(i + 2) % 4];
}
pic[x][y] += dir[i];
pic[tx][ty] -= dir[i];
vis[tx][ty] = 0;
}
}
}
return false;
}
int main(){
int sx, sy;
while(scanf("%d%d", &sx, &sy) == 2){
if(!sx && !sy) return 0;
memset(ans, 0, sizeof ans);
for(int i = 1; i <= 4; ++i){
for(int j = 1; j <= 6; ++j){
scanf("%d", &pic[i][j]);
}
}
for(maxn = 1; ; ++maxn){
memset(vis, 0, sizeof vis);
vis[sy][sx] = 1;
if(dfs(sy, sx, 0)){
for(int i = 0; i < maxn; ++i){
printf("%c", s[ans[i]]);
}
printf("\n");
break;
}
}
}
return 0;
}

  

UVA - 10384 The Wall Pusher(推门游戏)(IDA*)的更多相关文章

  1. UVa 10384 - The Wall Pushers

    链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  2. JavaScript写一个小乌龟推箱子游戏

    推箱子游戏是老游戏了, 网上有各种各样的版本, 说下推箱子游戏的简单实现,以及我找到的一些参考视频和实例: 推箱子游戏的在线DEMO : 打开 如下是效果图: 这个拖箱子游戏做了移动端的适配, 我使用 ...

  3. 用HTML5+原生js实现的推箱子游戏

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  4. 【CCpp程序设计2017】推箱子游戏

    我的还……支持撤销!用链表实现! 题目:推箱子小游戏(基于console) 功能要求: 将p09迷宫游戏改造为“推箱子”游戏: 在地图中增加箱子.箱子目标位置等图形: 当玩家将所有箱子归位,则显示玩家 ...

  5. JavaScript 推箱子游戏

    推箱子游戏的 逻辑非常简单,但是如果不动手的话,还是不太清楚.我在这里讲一下自己的思路. 制作推箱子,首先要有自己的设计素材.如下我也是网上找的素材 第二步,理清游戏的规则. 游戏规则: 1.小人将箱 ...

  6. three.js 制作一个三维的推箱子游戏

    今天郭先生发现大家更喜欢看我发的three.js小作品,今天我就发一个3d版本推箱子的游戏,其实webGL有很多框架,three.js并不合适做游戏引擎,但是可以尝试一些小游戏.在线案例请点击博客原文 ...

  7. C# 推箱子游戏&对战游戏

    推箱子游戏提纲,只有向右向上的操作,向左向下同理,后期需完善. namespace 推箱子 { class Program { static void Main(string[] args) { // ...

  8. 用C写一个简单的推箱子游戏(二)

    下面接着上一篇随笔<用C写一个简单的推箱子游戏(一)>来写 tuidong()函数是用来判断游戏人物前方情况的函数,是推箱子游戏中非常重要的一个函数,下面从它开始继续介绍推箱子的小程序怎么 ...

  9. 用C写一个简单的推箱子游戏(一)

    我现在在读大二,我们有一门课程叫<操作系统>,课程考查要求我们可以写一段程序或者写Windows.iOS.Mac的发展历程.后面我结合网上的资料参考,就想用自己之前简单学过的C写一关的推箱 ...

随机推荐

  1. 检测皮肤PH值、感知你的便意,健康是可穿戴设备的新风口?

    在经历最初的喧嚣与疯狂后,可穿戴设备近年来有些低调和沉寂.换句话说,虽然可穿戴设备销量在持续走高,但从形态和功能上,呈现出高度一致性.这似乎也在证明着,可穿戴设备已成为寻常可见的普通产品而已.不过在迈 ...

  2. 如果不想在django 的settings中保存mysql数据库的密码

    如题,你可以编写一个配置文件,用'OPTIONS' 将该配置文件导入进来,这样你发布到git上的源码上就没有你的数据库密码了. 这是django推荐的方法. # settings.py DATABAS ...

  3. 关于fpga的后仿真重要性

    也许你天天做些fpga,写完代码就直接编译成功,锁定引脚后,马上使用signaltaII软件. 也许你一天,你发现signaltapII看信号的痛苦,一个源代码文件修改一点,要花个20分钟编译一次. ...

  4. mysql dump 完全备

    创建表: MariaDB [xuegod]> create database xuegod; MariaDB [xuegod]> use xuegod; MariaDB [xuegod]& ...

  5. LoadRunner监控Linux系统

    需要下载3个包:  地址链接:链接:https://pan.baidu.com/s/1lltAa6JnjJ7Mr88duixUSQ 密码:5yiw(1)rsh-0.17-14.i386.rpm (2) ...

  6. Myeclipse项目出现红叉解决方案

    1.右键点击你的项目.选中properties 2.选中MyEclipse下的Project Facets里面的java 此时的版本号为1.5,修改 3.选中MyEclipse下的Project Fa ...

  7. MySQL学习之SQL基础(一)DDL

    Sql基础 DDL (data defination language) 创建表 CREATE TABLE emp( ename varchar(10), hiredate date, sal dec ...

  8. 未知进程问题,process information unavailable

    执行jps,有些未知进程: 2690 -- process information unavailable2666 原因:内存不足. cd /tmp/hsperfdata_impala/ 执行ll后, ...

  9. etcd入门

    简介 etcd是CoreOS团队于2013年6月发起的开源项目,它的目标是构建一个高可用的分布式键值(key-value)数据库. etcd内部采用raft协议作为一致性算法,基于Go语言实现. et ...

  10. IDEA快速升级模块版本号

    使用场景 一个多模块的项目中,在功能用重大更新后,需要升级版本号,如果不使用工具,需要手动更改每个pom.xml文件,而使用工具,就可以非常快速的完成版本号的更改. 基本步骤 0.  idea执行ma ...