问题描述

推箱子是一款经典的小游戏。游戏要求玩家将若干个箱子推到指定的位置,并以箱子移动次数最少作为目标。 现在,我们只需要考虑一个简化的版本——只有一个箱子。对于一张给定的地图,你需要判断是否可以将箱子推到目标位置,如果可以,你还需要求出箱子最少的移动次数。

输入格式

第一行有两个用单个空格分隔的正整数n,m(n,m<=100),表示输入一张n∗m的地图。 接下来n行,每行m个字母,字母分别是S, M, P, K, w,意义如下: S – 表示地图中的墙 M – 表示玩家的初始位置 P – 表示箱子的初始位置 K- 表示箱子的目标位置 w – 表示地图中的空格 M, P 和 K 在文件中只出现一次

输出格式

如果箱子不能移动到目的位置,则输出NO。

如果箱子能移动到目的位置,则输出最小的移动次数。

样例输入

10 12
SSSSSSSSSSSS
SwwwwwwwSSSS
SwSSSSwwSSSS
SwSSSSwwSKSS
SwSSSSwwSwSS
SwwwwwPwwwww
SSSSSSSwSwSw
SSSSSSMwSwww
SSSSSSSSSSSS
SSSSSSSSSSSS

样例输出

7

数据范围

对于10%的数据: n=1或m=1

另10%的数据: n=5,m=100或n=100,m=5

对于30%的数据: n,m<=50

对于100%的数据: n,m<=100



题目大意大概就是这样

这个题也算是祖传题了

上一届的yveh学长当时这题就没能做出来

点名批评某助教

大概讲一下思路以及心路历程

完整的代码为了(防止学弟学妹们借鉴,就不放了)

毕竟是祖传题嘛!!

大概主要是对箱子进行bfs,然后每一个点的状态存储当前的x,当前的y,当前的s(已走路程),还有上一次移动推过来的方向dir。

struct Node{
int x,y,s,dir;
};

然后每次对箱子bfs之前,都要对人进行bfs,看看能不能到箱子的四个位置,这时候上一次的dir就起作用了,可以知道人的初始位置

于是我就这样做了

不幸地TLE了

之后询问了以下周围同学,然后优化了一些步骤的先后顺序

So——速度大概快了负的百分之一吧!

我太难了

之后我就尝试了一些黑科技

强行加上头文件的O2O3Ofast优化之类的

可参考我的另一篇文章

卡常的编译命令(含O2优化)

当然还是没用啦

之后班里面的Hz同学告诉我关于vis数组的判断可以优化的方法

like this:

多次要用到vis[]时
可以用把vis改成int型,然后每次使用cnt++,如果cnt==vis[],说明此次已经遍历到了;
反之,就让vis[]=cnt;

速度好像是快了那么一点点点点

然而还是没啥用

然后,然后……

又get到一个技巧,就是每次对人进行bfs的时候,不要bfs四个方向,而是一次bfs;

因为起点都是一样的嘛,就是终点不太一样

然后一次bfs全图,再判断这个点能不能到

理论上这个可以让程序快四倍

我又交了

——————又T了

95分真的是气死人

不过我们班好多同学都卡在了95分上面;

最后的最后

我因为以前一直用的是STL里面的queue,毕竟我觉得更加方便,可以用结构体

现在——————不得不改成用数组表示队列了

确实我知道这样要快一点,但是基本上这要把整个程序都改一遍



又能咋办呢?

最后 费了好长一段时间,总算改完了

(如果不算之前的尝试吧)1Y !!!

小作业之后————感叹————STL害人!!!

真的不知道我什么时候又会真香

真香!

下面贴一点代码

关于最后表示队列的数组

int Q1[410000],Q2[410000],Q3[410000],Q4[410000],L,R;
int qx[410000],qy[410000],l,r;

箱子的bfs

void bfs(){
L=0,R=0;
for(int i=0;i<4;++i){
int nowx = th.x;
int nowy = th.y;
int nows=0;
int nowdir=i;
if(getbfs(my.x,my.y,nowx-dx[i],nowy-dy[i],th.x,th.y)){
R++;
Q1[R]=nowx,Q2[R]=nowy,Q3[R]=nows,Q4[R]=nowdir;
vis[nowdir][nowx][nowy]=true;
}
}
while(L<R){
L++;
int nowx=Q1[L];
int nowy=Q2[L];
int nows=Q3[L];
int nowdir=Q4[L];
BFS(nowx-dx[nowdir],nowy-dy[nowdir],nowx,nowy);
for(int i=0;i<4;++i){
int newsx=nowx+dx[i];
int newsy=nowy+dy[i];
int newss=nows+1;
int newsdir=i;
if(v[nowx-dx[i]][nowy-dy[i]]!=cnt) continue;
if(Map[newsx][newsy]=='S'||newsx<1||newsx>n||newsy<1||newsy>m||vis[i][newsx][newsy]==true) continue;
if(Map[newsx][newsy]=='K'){
flag=newss;
return;
}
R++;
Q1[R]=newsx,Q2[R]=newsy,Q3[R]=newss,Q4[R]=newsdir;
vis[i][newsx][newsy]=true;
}
}
return;
}

人的bfs

void BFS(const int &x,const int &y,const int &tx,const int &ty){
cnt++;
l=0,r=1;
qx[r]=x,qy[r]=y;
v[x][y]=cnt;
while(l<r){
l++;
int nowx=qx[l],nowy=qy[l];
for(int i=0;i<4;++i){
int newsx=nowx+dx[i];
int newsy=nowy+dy[i];
if(Map[newsx][newsy]=='S'||(newsx==tx&&newsy==ty)||newsx<1||newsx>n||newsy<1||newsy>m) continue;
if(v[newsx][newsy]==cnt) continue;
else{
v[newsx][newsy]=cnt;
r++;
qx[r]=newsx,qy[r]=newsy;
}
}
}
}

That's all.

SJTU T4143 推箱子的更多相关文章

  1. OC推箱子

    #include<stdio.h> #include<stdlib.h> int main(void) { char sr;//存储用户输入的指令 //绘制地图 char a[ ...

  2. c语言游戏推箱子

    前两天做了推箱子小游戏,看似简单的一个小游戏背后却 有巨大的秘密,这秘密就是一大堆逻辑. 自从学习了函数过后,的确是解决了很多问题,而且调用很方便,尽管我现在都不是很会调用. 写完一个函数,准备测试一 ...

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

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

  4. 用C#制作推箱子小游戏

    思路分析: 一.制作一个地图 二.地图中放置墙.箱子.人.目标等 三.让小人动起来完成推箱子动作 游戏制作: 1.按照上述地图制作一个地图  (12行×13列) 地图可以看做是行和列组成的,即可以看做 ...

  5. hdu.1254.推箱子(bfs + 优先队列)

    推箱子 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submis ...

  6. [游戏模版17] Win32 推箱子 迷宫

    >_<:Here introduce a simple game: >_<:resource >_<:only can push a box and finally ...

  7. [转]Flash ActionScript2.0面向对象游戏开发-推箱子

    本文转自:http://www.alixixi.com/Dev/W3C/Flash/2007/2007070868666.html 概述: Flash ActionScript2.0是一种面向对向的编 ...

  8. c#部分---网吧充值系统;简易的闹钟;出租车计费;简单计算器;对战游戏;等额本金法计算贷款还款利息等;随机生成10个不重复的50以内的整数;推箱子;

    网吧充值系统namespace ConsoleApplication1 { class Program { struct huiyuan { public string name; public st ...

  9. c++、c实现推箱子小游戏

    经过四次的修改和优化,终于将推箱子这个游戏完整的写出来了,今天就像大家分享一下这个游戏的编写. 这个游戏界面的编写总的来说不困难,主要是推动箱子的算法. (1)利用数组和windows api 即可写 ...

随机推荐

  1. python面试题整理(一)

    python基础:1.列表生成式和生成器表达式有什么区别 我说的是首先写法不一样,列表生成式用[],生成器表达式用(),其次列表生成是一次性生成一个完整的列表,生成器表达式返回的是一个一个的值,占用内 ...

  2. C++寒假作业2

    这个作业属于哪个课程 https://edu.cnblogs.com/campus/fzu/2020OOP 这个作业要求在哪里 https://edu.cnblogs.com/campus/fzu/2 ...

  3. Java 统计整数二进制中1的个数

    package cookie; public class CountBinary_1 { public static void main(String[] args) { System.out.pri ...

  4. 经典SQL50句

    50个常用的sql语句 Student(S#,Sname,Sage,Ssex) 学生表 Course(C#,Cname,T#) 课程表 SC(S#,C#,score) 成绩表 Teacher(T#,T ...

  5. mybatis今年笔记

    1.读取配置文件:用的就是解析Xml文件的技术 2.mybatis是支持自己写dao层的,但是没有必要. mybatis做的事情: 第一个创建代理对象,第二个在代理对象中调用方法. 3.相同的注解如果 ...

  6. prometheus配置简介

    参考网页:https://my.oschina.net/wangyunlong/blog/3060776 global: scrape_interval:             15s evalua ...

  7. SVM手撕公式

    卓越源于坚持,努力须有方向. 如上图所示,有一堆训练数据的正负样本,标记为:,假设有一个超平面H:,可以把这些样本正确无误地分割开来,同时存在两个平行于H的超平面H1和H2: 使离H最近的正负样本刚好 ...

  8. 读《软件需求最佳实践》YOUGAN

    这几天在看<软件需求最佳实践>作者徐锋老师的软件需求培训,三天的课程,虽然原来对需求也关注了很多,自己也做过需求分析和开发的工作,但是这次培训感觉收获还是很多.三天的培训先做个记录,后续多 ...

  9. 2020PHP面试-PHP篇

    一.列举一些PHP的设计模式 单例模式:保证在整个应用程序的生命周期中,任何一个时刻,单例类的实例都只存在一个,同时这个类还必须提供一个访问该类的全局访问点. 工厂模式:定义一个创建对象的接口,但是让 ...

  10. [GXYCTF2019]Ping Ping Ping

    0x00 知识点 命令执行变量拼接 /?ip=127.0.0.1;a=g;cat$IFS$1fla$a.php 过滤bash用sh执行 echo$IFS$1Y2F0IGZsYWcucGhw|base6 ...