【Luogu】P1312Mayan游戏(暴搜)
由于是暴搜题,所以这篇博客只讲怎么优化剪枝,以及一些细节。
模拟消除思路:因为消除可以拆分成小的横条或竖条,而这些条的长度至少为三,所以一块可消除的区域至少会有一个中心点。这里的中心点可以不在正中间,只需要不是条上的第一个或者最后一个。
于是枚举中间点,搜索它为中心最多向四个方向能扩展多远。如果搜索出来的横向满足长度要求,或竖向满足长度要求,就给他们打上一个标记。
注意,这里只是打上标记,不能直接清零,很可能另一个方块的结算还得用到这个方块。
等到枚举所有的中间点并给所有可消除的方块打上标记之后,可以把所有标记上的方块清空。然后检查有没有地方可以落下去。
注意所有方块落下去之后还有可能接着消消乐,所以在落完之后还要再循环回去检查有没有可消除的方块。这里使用一个递归的代码来实现。
int count(int x){
bool vis[][],flag=;
int cnt=;
memset(vis,,sizeof(vis));
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j){
int col=Map[x][i][j];
if(!col) continue;
int u=i,d=i,l=j,r=j;
while(u>&&Map[x][u-][j]==col) u--;
while(d<n&&Map[x][d+][j]==col) d++;
while(l>&&Map[x][i][l-]==col) l--;
while(r<m&&Map[x][i][r+]==col) r++;
if(d-u>=)
for(int k=u;k<=d;++k) vis[k][j]=;
if(r-l>=)
for(int k=l;k<=r;++k) vis[i][k]=;
}
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j)
if(vis[i][j]){
Map[x][i][j]=; cnt++;
}
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j)
if(!Map[x][i][j]){
int s=i;
while(s<n&&!Map[x][s][j]) s++;
if(!Map[x][s][j]) continue;
flag=;
Map[x][i][j]=Map[x][s][j];
Map[x][s][j]=;
}
if(flag) cnt+=count(x);
return cnt;
}
注意最后倒数第三行。flag表示的是有没有方块落下,因为有落下方块就有新一轮消除的可能性,所以可以递归这个函数,直到没有任何方块落下为止。此时递归终止,开始回溯计算答案。
再说说剪枝的几个小技巧。
1、优先考虑坐标字典序小的向右移动,这样一旦搜到答案就是字典序最小的解。
2、只有当左边没有方块的时候才向左移动,否则右面方块向左移动等价于左面方块向右移动,而这个状态已经搜过了。
3、不交换两个颜色相同的方块。这个没什么好说的。
最后给出代码。
#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<cctype>
#include<cstring>
using namespace std; inline long long read(){
long long num=,f=;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-') f=-;
ch=getchar();
}
while(isdigit(ch)){
num=num*+ch-'';
ch=getchar();
}
return num*f;
} int num;
int n=,m=;
int Max;
int Map[][][];
int posx[],posy[],move[]; inline void copy(int x){
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j) Map[x+][i][j]=Map[x][i][j];
} int count(int x){
bool vis[][],flag=;
int cnt=;
memset(vis,,sizeof(vis));
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j){
int col=Map[x][i][j];
if(!col) continue;
int u=i,d=i,l=j,r=j;
while(u>&&Map[x][u-][j]==col) u--;
while(d<n&&Map[x][d+][j]==col) d++;
while(l>&&Map[x][i][l-]==col) l--;
while(r<m&&Map[x][i][r+]==col) r++;
if(d-u>=)
for(int k=u;k<=d;++k) vis[k][j]=;
if(r-l>=)
for(int k=l;k<=r;++k) vis[i][k]=;
}
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j)
if(vis[i][j]){
Map[x][i][j]=; cnt++;
}
for(register int i=;i<=n;++i)
for(register int j=;j<=m;++j)
if(!Map[x][i][j]){
int s=i;
while(s<n&&!Map[x][s][j]) s++;
if(!Map[x][s][j]) continue;
flag=;
Map[x][i][j]=Map[x][s][j];
Map[x][s][j]=;
}
if(flag) cnt+=count(x);
return cnt;
} void dfs(int use,int deep){
if(use!=&&deep==Max) return;
if(use==){
if(deep==Max){
for(int i=;i<=Max;++i) printf("%d %d %d\n",posy[i]-,posx[i]-,move[i]);
exit();
}
return;
}
copy(deep);
for(register int j=;j<=m;++j)
for(register int i=;i<=n;++i){
if(!Map[deep][i][j]) continue;
if(Map[deep][i][j]!=Map[deep][i][j+]&&j<m){
int a=Map[deep][i][j],b=Map[deep][i][j+];
Map[deep+][i][j]=b;Map[deep+][i][j+]=a;
posx[deep+]=i; posy[deep+]=j; move[deep+]=;
int q=count(deep+);
dfs(use-q,deep+);
copy(deep);
}
if(!Map[deep][i][j-]&&j>){
int a=Map[deep][i][j],b=Map[deep][i][j-];
Map[deep+][i][j]=b;Map[deep+][i][j-]=a;
posx[deep+]=i; posy[deep+]=j; move[deep+]=-;
int s=i;
while(s>&&Map[deep+][s-][j-]==){
Map[deep+][s--][j-]=;
Map[deep+][s][j-]=a;
}
int q=count(deep+);
dfs(use-q,deep+);
copy(deep);
}
}
} int main(){
Max=read();
int start=;
for(int i=;i<=m;++i)
for(int j=;;j++){
Map[][j][i]=read();
if(Map[][j][i]==) break;
start++;
}
dfs(start,);
printf("-1");
return ;
}
【Luogu】P1312Mayan游戏(暴搜)的更多相关文章
- 洛谷 1312 Mayan游戏——暴搜+剪枝
题目:https://www.luogu.org/problemnew/show/P1312 自己写了很久.又T又WA的. 发现对题理解有误.改完后应该只有T了,但还是T的. 自己写了许多剪枝,很鸡肋 ...
- NOIP 2011 Mayan游戏 大暴搜
题目链接:https://www.luogu.org/problemnew/show/P1312 我的第一篇题解!! 当然感谢ZAGER 的提示,他的链接https://www.cnblogs.com ...
- 【BZOJ-3033】太鼓达人 欧拉图 + 暴搜
3033: 太鼓达人 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 204 Solved: 154[Submit][Status][Discuss] ...
- c++20701除法(刘汝佳1、2册第七章,暴搜解决)
20701除法 难度级别: B: 编程语言:不限:运行时间限制:1000ms: 运行空间限制:51200KB: 代码长度限制:2000000B 试题描述 输入正整数n,按从小到大的顺序输出所有 ...
- Codeforces Round #238 (Div. 2) D. Toy Sum 暴搜
题目链接: 题目 D. Toy Sum time limit per test:1 second memory limit per test:256 megabytes 问题描述 Little Chr ...
- poj 3080 Blue Jeans(水题 暴搜)
题目:http://poj.org/problem?id=3080 水题,暴搜 #include <iostream> #include<cstdio> #include< ...
- Sicily1317-Sudoku-位运算暴搜
最终代码地址:https://github.com/laiy/Datastructure-Algorithm/blob/master/sicily/1317.c 这题博主刷了1天,不是为了做出来,AC ...
- codeforces 339C Xenia and Weights(dp或暴搜)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud Xenia and Weights Xenia has a set of weig ...
- Usaco 2.3 Zero Sums(回溯DFS)--暴搜
Zero SumConsider the sequence of digits from 1 through N (where N=9) in increasing order: 1 2 3 ... ...
随机推荐
- 【数据分析 R语言实战】学习笔记 第十一章 对应分析
11.2对应分析 在很多情况下,我们所关心的不仅仅是行或列变量本身,而是行变量和列变量的相互关系,这就是因子分析等方法无法解释的了.1970年法国统计学家J.P.Benzenci提出对应分析,也称关联 ...
- MySQL索引使用等
- codevs 1131 统计单词数 2011年NOIP全国联赛普及组
时间限制: 1 s 空间限制: 128000 KB 题目等级 : 白银 Silver 题目描述 Description 一般的文本编辑器都有查找单词的功能,该功能可以快速定位特定单词在文章中的位 ...
- (转)MyBatis框架的学习(六)——MyBatis整合Spring
http://blog.csdn.net/yerenyuan_pku/article/details/71904315 本文将手把手教你如何使用MyBatis整合Spring,这儿,我本人使用的MyB ...
- poj2104 K大数 划分树
题意:给定一个数列,求一个区间的第K大数 模板题, 其中的newl, newr 有点不明白. #include <iostream> #include <algorithm> ...
- caffe修改需要的东西 6:40
https://blog.csdn.net/zhaishengfu/article/details/51971768?locationNum=3&fps=1
- python打开.pkl的文件并显示里面的内容
pkl文件是pyhthon里面保存文件的一种格式,如果直接打开会显示一堆序列化的东西.正确的打开方式如下: import cPickle as pickle f = open('path') info ...
- Delphi与JAVA互加解密AES算法
搞了半天终于把这个对应的参数搞上了,话不多说,先干上代码: package com.bss.util; import java.io.UnsupportedEncodingException; imp ...
- 三. python面向对象
第七章.面向对象基础 1.面向对象基础 类和对象: a. 创建类 class 类名: def 方法名(self,xxx): pass b. 创建对象 对象 = 类名() c. 通过对象执行方法 对象. ...
- 控制器生命周期方法(LifeCycle)
1.init方法: 在init方法中实例化必要的对象(遵从LazyLoad思想) init方法中初始化ViewController本身 2.loadView方法: 当view需要被展示而它 ...