其实就是一道锻炼码力的简单题……

看到题目中的\(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游戏的更多相关文章

  1. Luogu P1312 Mayan游戏(搜索)

    P1312 Mayan游戏 题意 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个\(7\)行\(\times 5\)列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必 ...

  2. [题目] Luogu P1312 Mayan游戏

    题面 题目描述 $ Mayan puzzle $是最近流行起来的一个游戏.游戏界面是一个 \(7行 \times 5列\)的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放 ...

  3. 洛谷P1312 Mayan游戏

    P1312 Mayan游戏 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他 ...

  4. [NOIP2011] 提高组 洛谷P1312 Mayan游戏

    题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定 ...

  5. P1312 Mayan游戏

    题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定 ...

  6. 洛谷 P1312 Mayan游戏

    题解:搜索+模拟 剪枝: 最优性剪枝:x从小到大,y从小到大,第一次搜到的就是字典序最小 的最优解. 最优性剪枝:把一个格子和左边格子交换,和左边格子和右边格 子交换是等价的,显然让左边格子和右边交换 ...

  7. 洛古 P1312 Mayan游戏(dfs+剪枝)

    题目链接 这道题和俄罗斯方块很像 很明显,我们可以看出这是一个dfs,但是,我们需要几条剪枝: 1.如果只剩下1个或2个同样颜色的方块,那么直接退出 2.相同的块不用交换 3.注意优先性,优先左边换右 ...

  8. [Luogu 1312] noip11 Mayan游戏

    [Luogu 1312] noip11 Mayan游戏 Problem: Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即 ...

  9. 洛谷P1312 [NOIP2011提高组Day1T3]Mayan游戏

    Mayan游戏 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游 ...

随机推荐

  1. 【LeetCode】109. Convert Sorted List to Binary Search Tree 解题报告(Python)

    [LeetCode]109. Convert Sorted List to Binary Search Tree 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id ...

  2. 【LeetCode】824. Goat Latin 解题报告(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 日期 题目地址:https://leetcode.c ...

  3. 【模型推理】量化实现分享三:详解 ACIQ 对称量化算法实现

      欢迎关注我的公众号 [极智视界],回复001获取Google编程规范   O_o   >_<   o_O   O_o   ~_~   o_O   大家好,我是极智视界,本文剖析一下AC ...

  4. Spring中的@Bean注解

    @Bean 基础概念 @Bean:Spring的@Bean注解用于告诉方法,产生一个Bean对象,然后这个Bean对象交给Spring管理.产生这个Bean对象的方法Spring只会调用一次,随后这个 ...

  5. Java实习生常规技术面试题每日十题Java基础(一)

    目录 1.Java 的 "一次编写,处处运行"如何实现? 2.描述JVM运行原理. 3.为什么Java没有全局变量? 4.说明一下public static void main(S ...

  6. 什么是NaN?它的类型是什么?如何可靠的测试一个值是否等于NaN?

    NaN属性表示"不是数字"的值.这个特殊值是由于一个操作数是非数字的(例如"abc"/4)或者因为操作的结果是非数字而无法执行的. 虽然看起来很简单,但是NaN ...

  7. MySQL8.0.20安装详解

    https://blog.csdn.net/yeb112233/article/details/106042867/ alter user root@localhost identified by ' ...

  8. Drools集成SpringBootStarter

    1.说明 基于fast-drools-spring-boot-starter, 能够方便的将规则引擎Drools集成到Spring Boot, 基于前面介绍过的文章Drools集成SpringBoot ...

  9. Linux下设置普通用户使用sudo命令

    1.登录root用户 2.增加root用户对文件sudoers的写权限 chmod u+w /etc/sudoers 3.编辑sudoers,把用户mysql添加进去 vi /etc/sudoers ...

  10. js 模块化 -- export 时 一个默认和多个默认的写法

    js文件 只有一个  food 类,一般写 export {food} 或者 export {food as default} 那么如果有多个呢? 这样 export {food, food2} 或者 ...