[luogu P1312]Mayan游戏
其实就是一道锻炼码力的简单题……
看到题目中的\(0<x\leqslant 5\)也就知道是爆搜了吧(
我们仿照写游戏的方法多写几个函数,能够有效降低错误率(确信
我们写出大致的搜索流程来:
如果当前步数大于\(n\)直接返回;
如果当前已经为终态,直接输出答案;
否则枚举每种可能情况继续搜。
没错就是这么暴力
接下来讲具体实现。
首先我们对全局变量和数组进行一下约定:
int n;//见题目
int board[10][10];//棋盘
int step[10][10];//存储步数
int mem[10][10][10];//鉴于搜索下一步的时候我们会不可避免地更改原数组,我们需要保存一下当前的状态
bool del[10][10];//见remove函数部分
一开始的读入非常简单:读到0换行即可
scanf("%d",&n);
for(int i=1;i<=5;i++)
{
int cnt=0;
while(1)
{
int xx;scanf("%d",&xx);
if(xx==0)break;
board[i][++cnt]=xx;
}
}
我们需要对题目的一些操作进行一下模拟。
首先是fall函数:它让悬空的块下落。
可以发现,我们用一个变量来存下移的最终位置,就可以简单地进行实现。
void fall()
{
int cnt;
for(int i=1;i<=5;i++)
{
cnt=0;
for(int j=1;j<=7;j++)
{
if(board[i][j]==0)cnt++;
else
{
if(cnt==0)continue;
board[i][j-cnt]=board[i][j];
board[i][j]=0;
}
}
}
}
然后是三消remove函数:
我们分别暴力判断横向和纵向每一次可以消掉哪些块,全都用del数组给标记出来,最后一起消掉。
别忘了最后要将del数组清零,消完之后fall一次让可能悬空的块下坠。
当然就像样例一样,一次消除是:消除当前->下坠->出现新的可三消块->再次消除当前……
于是我们将这个函数修改一下,让它发现本次有消除操作就返回1,没有就返回0。
于是我们可以这样调用remove函数:
while(remove());
简单粗暴(
remove函数实现:
bool remove()
{
bool flag=0;
for(int i=1;i<=5;i++)
for(int j=2;j<=6;j++)
if(board[i][j]!=0&&board[i][j]==board[i][j-1]&&board[i][j]==board[i][j+1])
{
flag=1;
del[i][j]=del[i][j-1]=del[i][j+1]=1;
}
for(int i=2;i<=4;i++)
for(int j=1;j<=7;j++)
if(board[i][j]!=0&&board[i][j]==board[i-1][j]&&board[i][j]==board[i+1][j])
{
flag=1;
del[i][j]=del[i-1][j]=del[i+1][j]=1;
}
if(!flag)return 0;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
if(del[i][j])
board[i][j]=del[i][j]=0;
fall();
return 1;
}
接下来是移动块函数move和判断是否结束函数gameover。
move函数只需要移动一次并调用fall坠落一次,然后remove即可。
gameover直接全屏扫。
void move(int xx,int yy,int dir)//用dir标记方向
{
swap(board[xx][yy],board[xx+dir][yy]);
fall();while(remove());
}
bool gameover()
{
for(int i=1;i<=5;i++)
if(board[i][1]!=0)
return 0;
return 1;
}
接下来是dfs函数的枚举部分。
首先我们用mem储存一下;
然后只需要分方向枚举,枚举到一种情况就记录步数继续搜;
回溯的时候撤销所记录的步数,用mem数组换回原来的状态即可。
一个小剪枝:在交换之前判断一下,避免将相同的块交换即可。
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
mem[xx][i][j]=board[i][j];
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
if(board[i][j]!=0)
{
if(i<=4&&board[i][j]!=board[i+1][j])
{
move(i,j,1);
step[xx][1]=i-1;step[xx][2]=j-1;step[xx][3]=1;
dfs(xx+1);
step[xx][1]=step[xx][2]=step[xx][3]=-1;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
board[i][j]=mem[xx][i][j];
}
if(i>=2&&board[i][j]!=board[i-1][j])
{
move(i,j,-1);
step[xx][1]=i-1;step[xx][2]=j-1;step[xx][3]=-1;
dfs(xx+1);
step[xx][1]=step[xx][2]=step[xx][3]=-1;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
board[i][j]=mem[xx][i][j];
}
}
于是我们就这么水完了一道蓝题
完整代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
int n,board[10][10],step[10][10],mem[10][10][10];
bool del[10][10];
void fall()
{
int cnt;
for(int i=1;i<=5;i++)
{
cnt=0;
for(int j=1;j<=7;j++)
{
if(board[i][j]==0)cnt++;
else
{
if(cnt==0)continue;
board[i][j-cnt]=board[i][j];
board[i][j]=0;
}
}
}
}
bool remove()
{
bool flag=0;
for(int i=1;i<=5;i++)
for(int j=2;j<=6;j++)
if(board[i][j]!=0&&board[i][j]==board[i][j-1]&&board[i][j]==board[i][j+1])
{
flag=1;
del[i][j]=del[i][j-1]=del[i][j+1]=1;
}
for(int i=2;i<=4;i++)
for(int j=1;j<=7;j++)
if(board[i][j]!=0&&board[i][j]==board[i-1][j]&&board[i][j]==board[i+1][j])
{
flag=1;
del[i][j]=del[i-1][j]=del[i+1][j]=1;
}
if(!flag)return 0;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
if(del[i][j])
board[i][j]=del[i][j]=0;
fall();
return 1;
}
void move(int xx,int yy,int dir)
{
swap(board[xx][yy],board[xx+dir][yy]);
fall();while(remove());
}
bool gameover()
{
for(int i=1;i<=5;i++)
if(board[i][1]!=0)
return 0;
return 1;
}
void dfs(int xx)
{
if(gameover())
{
for(int i=1;i<=n;i++)
{
for(int j=1;j<=3;j++)printf("%d ",step[i][j]);
printf("\n");
}
exit(0);
}
if(xx>n)return;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
mem[xx][i][j]=board[i][j];
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
if(board[i][j]!=0)
{
if(i<=4&&board[i][j]!=board[i+1][j])
{
move(i,j,1);
step[xx][1]=i-1;step[xx][2]=j-1;step[xx][3]=1;
dfs(xx+1);
step[xx][1]=step[xx][2]=step[xx][3]=-1;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
board[i][j]=mem[xx][i][j];
}
if(i>=2&&board[i][j]!=board[i-1][j])
{
move(i,j,-1);
step[xx][1]=i-1;step[xx][2]=j-1;step[xx][3]=-1;
dfs(xx+1);
step[xx][1]=step[xx][2]=step[xx][3]=-1;
for(int i=1;i<=5;i++)
for(int j=1;j<=7;j++)
board[i][j]=mem[xx][i][j];
}
}
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=5;i++)
{
int cnt=0;
while(1)
{
int xx;scanf("%d",&xx);
if(xx==0)break;
board[i][++cnt]=xx;
}
}
dfs(1);
printf("-1\n");
return 0;
}
最慢的点跑了1.09s,但还是够AC了(笑
[luogu P1312]Mayan游戏的更多相关文章
- Luogu P1312 Mayan游戏(搜索)
P1312 Mayan游戏 题意 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个\(7\)行\(\times 5\)列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必 ...
- [题目] Luogu P1312 Mayan游戏
题面 题目描述 $ Mayan puzzle $是最近流行起来的一个游戏.游戏界面是一个 \(7行 \times 5列\)的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放 ...
- 洛谷P1312 Mayan游戏
P1312 Mayan游戏 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他 ...
- [NOIP2011] 提高组 洛谷P1312 Mayan游戏
题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定 ...
- P1312 Mayan游戏
题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定 ...
- 洛谷 P1312 Mayan游戏
题解:搜索+模拟 剪枝: 最优性剪枝:x从小到大,y从小到大,第一次搜到的就是字典序最小 的最优解. 最优性剪枝:把一个格子和左边格子交换,和左边格子和右边格 子交换是等价的,显然让左边格子和右边交换 ...
- 洛古 P1312 Mayan游戏(dfs+剪枝)
题目链接 这道题和俄罗斯方块很像 很明显,我们可以看出这是一个dfs,但是,我们需要几条剪枝: 1.如果只剩下1个或2个同样颜色的方块,那么直接退出 2.相同的块不用交换 3.注意优先性,优先左边换右 ...
- [Luogu 1312] noip11 Mayan游戏
[Luogu 1312] noip11 Mayan游戏 Problem: Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即 ...
- 洛谷P1312 [NOIP2011提高组Day1T3]Mayan游戏
Mayan游戏 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游 ...
随机推荐
- Collision(hdu5114)
Collision Time Limit: 15000/15000 MS (Java/Others) Memory Limit: 512000/512000 K (Java/Others)Tot ...
- 终于做了一把MySQL调参boy
本文通过笔者经历的一个真实案例来介绍一个MySQL中的重要参数innodb_buffer_pool_size,希望能给大家带来些许收获,当遇到类似性能问题时可以多一种思考方式. 图片拍摄于大唐不夜城 ...
- MCU变量加载过程
前言 在开发mcu代码的时候经常会有些疑惑,变量是怎么在编译之后进入单片机的ram区的呢,特别是在使用keil开发的时候.后来在接触gcc编译器和自研的mcu后,终于明白了这个问题.实际上变量编译后被 ...
- JSP中的九大内置对象
JSP九大内置对象 pageContext 存东西 Request 存东西 Response Session 存东西 Application(servletContext) 存东西 config(se ...
- Java实习生常规技术面试题每日十题Java基础(四)
目录 1.String 和StringBuffer的区别. 2.数组有没有length()这个方法? String有没有length()这个方法? 3.final, finally, finalize ...
- 图解MongoDB集群部署原理(3)
MongoDB的集群部署方案中有三类角色:实际数据存储结点.配置文件存储结点和路由接入结点. 连接的客户端直接与路由结点相连,从配置结点上查询数据,根据查询结果到实际的存储结点上查询和存储数据.Mon ...
- 深入 Laravel 内核之外观模式(门面模式)
门面模式核心内容: 客户端与子系统的通信通过外观对象进行: 外观对象封装一系列子系统的具体对应方法,对客户端只需暴露一个单一的入口方法: 客户端通过访问外观对象即可调用子系统的基础方法,无需关心子系统 ...
- elasticsearch设置密码
ELK - X-Pack设置用户密码 enable X-Pack security vi elasticsearch.yml #首先开启x-pack插件 xpack.security.enabled: ...
- vue中把一个事件绑定到子组件上
官网上是这样描述的 你可能有很多次想要在一个组件的根元素上直接监听一个原生事件.这时,你可以使用 v-on的 .native 修饰符 父组件App.vue <template> <d ...
- vue2.0多页面开发
我们平常用vue开发的时候总觉得vue好像就是专门为了单页面应用而诞生的,其实不是.因为vue在工程化开发的时候很依赖webpack,而webpack是将所有的资源整合到一块,弄成一个单页面.但是vu ...