洛谷 P1312 Mayan游戏
题解:搜索+模拟
剪枝:
最优性剪枝:x从小到大,y从小到大,第一次搜到的就是字典序最小
的最优解。
最优性剪枝:把一个格子和左边格子交换,和左边格子和右边格
子交换是等价的,显然让左边格子和右边交换更优。
可行性剪枝:如果当前格子某个颜色个数为1或者2return 一定消不去。
最优性剪枝:相同颜色格子交换并没有什么卵用,当左边是空时和左边交换
几个操作
(1)down()函数 目的是为了让腾空的格子落下
(2)xiao()函数 目的是让三个相同颜色的格子消去
(3)check()函数 当前颜色是否都被消去了
错因:不是蠢是弱呀...,down函数写错了,还有tmp[][]不能设成
全局变量,否则回溯不了....md...orz
代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define maxn 20
using namespace std;
int n;
int map[maxn][maxn],re[maxn][maxn],ans[maxn][maxn],cnt[maxn]; void read(){
scanf("%d",&n);
for(int i=;i<=;i++){
int nu=,x;
while(){
scanf("%d",&x);
if(!x)break;
map[i][++nu]=x;
}
}
} bool check(){
for(int i=;i<=;i++)
for(int j=;j<=;j++)
if(map[i][j])return false;
return true;
} void down(){
for(int i=;i<=;i++){
int nu=,t;
for(int j=;j<=;j++){
// if(map[i][j])map[i][++nu]=map[i][j],map[i][j]=0;
//留住上面沙茶的一行,最后map[][]竟然清0了、
if(map[i][j]){t=map[i][j];map[i][j]=;map[i][++nu]=t;}
}
}
} bool xiao(){
bool flag=false;
memset(re,,sizeof(re));
for(int i=;i<=;i++){
for(int j=;j<=;j++){
if(map[i][j]&&map[i][j]==map[i][j-]&&map[i][j]==map[i][j+])
flag=true,re[i][j]=re[i][j-]=re[i][j+]=true;
if(map[i][j]&&map[i][j]==map[i+][j]&&map[i][j]==map[i-][j])
flag=true,re[i][j]=re[i+][j]=re[i-][j]=true;
}
}
for(int i=;i<=;i++)
for(int j=;j<=;j++)
if(re[i][j])map[i][j]=;
return flag;
} void dfs(int x){
if(x==n+){
if(check()){
for(int i=;i<=n;i++)
printf("%d %d %d\n",ans[i][]-,ans[i][]-,ans[i][]);
exit();
}
return;
}
int tmp[][];memset(cnt,,sizeof(cnt));
for(int i=;i<=;i++)for(int j=;j<=;j++)tmp[i][j]=map[i][j],cnt[map[i][j]]++;
for(int i=;i<=;i++)if(cnt[i]==||cnt[i]==)return;
for(int i=;i<=;i++){
for(int j=;j<=;j++){
if(map[i][j]==)break;
if(map[i][j]!=map[i+][j]&&i!=){
swap(map[i][j],map[i+][j]);
ans[x][]=i,ans[x][]=j,ans[x][]=;
down();while(xiao())down();
dfs(x+);
for(int i=;i<=;i++)for(int j=;j<=;j++)map[i][j]=tmp[i][j];
}
if(map[i-][j]==&&i!=){
swap(map[i][j],map[i-][j]);
ans[x][]=i,ans[x][]=j,ans[x][]=-;
down();while(xiao())down();
dfs(x+);
for(int i=;i<=;i++)for(int j=;j<=;j++)map[i][j]=tmp[i][j];
}
}
}
} int main(){
read();dfs();
puts("-1");
return ;
}
AC
20180825
22:02:01
一年后凭借对题解的记忆又写了一遍....比去年好多了...
注意记录答案,如果用flag=1表示找到了答案return,
记录的答案有可能改变。
吸了氧终于过了
// luogu-judger-enable-o2
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std; int n; int flag; int a[][],ct[]; struct A
{
int x,y,d;
}ans[]; bool ok()
{
for(int i=;i<;i++)
{
for(int j=;j<;j++)
{
if(a[i][j]) return false;
}
}
return true;
} void drop()
{
for(int i=;i<;i++)
{
for(int j=;j<;j++)
{
int x=j,y=i;
if(a[j][i]&&(x->=&&!a[x-][y]))
{
while(x->=&&!a[x-][y]) x--;
a[x][y]=a[j][i];
a[j][i]=;
}
}
}
} bool clear()
{
int yes=;
int k[][];
memset(k,,sizeof(k));
for(int i=;i<;i++)
{
for(int j=;j<;j++)
{
if(!a[i][j]) continue;
if(j+<&&j->=&&a[i][j]==a[i][j+]&&a[i][j]==a[i][j-]) yes=,k[i][j]=k[i][j+]=k[i][j-]=;
if(i+<&&i->=&&a[i][j]==a[i+][j]&&a[i][j]==a[i-][j]) yes=,k[i][j]=k[i-][j]=k[i+][j]=;
}
}
for(int i=;i<;i++)
{
for(int j=;j<;j++)
{
if(k[i][j]) a[i][j]=;
}
}
return yes;
} void dfs(int now)
{
int b[][];
if(now==n+&&ok())
{
for(int i=;i<=n;i++) printf("%d %d %d\n",ans[i].x,ans[i].y,ans[i].d);
exit();
}
if(now>=n+) return;
memset(ct,,sizeof(ct));
for(int i=;i<;i++) for(int j=;j<;j++) b[i][j]=a[i][j],ct[b[i][j]]++;
for(int i=;i<=;i++) if(ct[i]&&ct[i]<) return ;
for(int i=;i<;i++)
{
for(int j=;j<;j++)
{
if(!a[j][i]) continue;
if(a[j][i]!=a[j][i+]&&i+<)
{
swap(a[j][i],a[j][i+]);
drop(); while(clear()) drop();
ans[now].x=i;ans[now].y=j;ans[now].d=;
dfs(now+);
for(int c=;c<;c++) for(int d=;d<;d++) a[c][d]=b[c][d];
}
if(i->=&&a[j][i]!=a[j][i-])
{
swap(a[j][i],a[j][i-]);
drop();while(clear()) drop();
ans[now].x=i;ans[now].y=j;ans[now].d=-;
dfs(now+);
for(int c=;c<;c++) for(int d=;d<;d++) a[c][d]=b[c][d];
}
}
}
} int main()
{
scanf("%d",&n);
for(int i=;i<;i++)
{
int cnt=,x;
while()
{
scanf("%d",&x);
if(!x) break;
a[cnt++][i]=x;
}
}
dfs();
printf("-1\n");
return ;
}
AC
洛谷 P1312 Mayan游戏的更多相关文章
- 洛谷P1312 Mayan游戏
P1312 Mayan游戏 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他 ...
- [NOIP2011] 提高组 洛谷P1312 Mayan游戏
题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游戏通关是指在规定 ...
- 洛谷 1312 Mayan游戏——暴搜+剪枝
题目:https://www.luogu.org/problemnew/show/P1312 自己写了很久.又T又WA的. 发现对题理解有误.改完后应该只有T了,但还是T的. 自己写了许多剪枝,很鸡肋 ...
- 洛谷1312 Mayan游戏
原题链接 讨厌这种大搜索题 基本就是模拟搜索,注意细节即可. 以下是我用的两个剪枝: 将块向左移的前提是左边为空,因为该题要求先右后左,所以若左边有块,那么在上一次搜索向右移的时候一定会搜过,且字典序 ...
- 洛古 P1312 Mayan游戏(dfs+剪枝)
题目链接 这道题和俄罗斯方块很像 很明显,我们可以看出这是一个dfs,但是,我们需要几条剪枝: 1.如果只剩下1个或2个同样颜色的方块,那么直接退出 2.相同的块不用交换 3.注意优先性,优先左边换右 ...
- 洛谷 P2197 nim游戏
洛谷 P2197 nim游戏 题目描述 甲,乙两个人玩Nim取石子游戏. nim游戏的规则是这样的:地上有n堆石子(每堆石子数量小于10000),每人每次可从任意一堆石子里取出任意多枚石子扔掉,可以取 ...
- 洛谷 P1965 转圈游戏
洛谷 P1965 转圈游戏 传送门 思路 每一轮第 0 号位置上的小伙伴顺时针走到第 m 号位置,第 1 号位置小伙伴走到第 m+1 号位置,--,依此类推,第n − m号位置上的小伙伴走到第 0 号 ...
- Luogu P1312 Mayan游戏(搜索)
P1312 Mayan游戏 题意 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个\(7\)行\(\times 5\)列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必 ...
- 洛谷P1312 [NOIP2011提高组Day1T3]Mayan游戏
Mayan游戏 题目描述 Mayan puzzle是最近流行起来的一个游戏.游戏界面是一个 7 行5 列的棋盘,上面堆放着一些方块,方块不能悬空堆放,即方块必须放在最下面一行,或者放在其他方块之上.游 ...
随机推荐
- Nginx 自定义404、500错误页面跳转
自定义Nginx错误界面跳转 1.开启Nginx.conf配置文件下的自定义接口参数. http { fastcgi_intercept_errors on; } 2.在Server区域添加自定义的错 ...
- Linux下捕捉信号
关于 信号signal的知识铺垫 点这里 信号由三种处理方式: 忽略 执行该信号的默认处理动作 捕捉信号 如果信号的处理动作是用户自定义函数,在信号递达时就调用这个自定义函数,这称为捕捉信号. 进程收 ...
- bzoj 3479: [Usaco2014 Mar]Watering the Fields
3479: [Usaco2014 Mar]Watering the Fields Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 174 Solved ...
- Java Override和@Override
Override : 重写. 当子类的某个方法的方法名.返回值.参数列表均与父类的方法保持一致,我们就可以说子类重写了父类的该方法. 其中需要注意: 父类中修饰符为private, static, f ...
- 关于log4j.properties例子:DailyRollingFileAppender
package com.v512.log4j; import org.apache.log4j.Logger; public class HelloLog4J { // 构造记录器,形参是记录器所在的 ...
- power shell添加vim
1.去Vim官网下载适合操作系统的可执行文件 地址:https://www.vim.org/download.php#pc 2.找到Vim文件夹中的vimrc文件进行修改,增加下面这4行. set e ...
- spring3: 切面及通知实例 Aspectj的aop
1.前置通知 接口: package chapter1.server; public interface IHelloService { public void sayAdvisorBefore(St ...
- redis、kafka、rabittMQ对比 (转)
本文不对三者之间的性能进行对比,只是从三者的特性上区分他们,并指出三者的不用应用场景. 1.publish/subscribe 发布订阅模式如下图所示可以具有多个生产者和发布者,redis.kafka ...
- python+mitmproxy抓包过滤+redis消息订阅+websocket实时消息发送,日志实时输出到web界面
本实例实现需求 在游戏SDK测试中,经常需要测试游戏中SDK的埋点日志是否接入正确.本实例通过抓包(客户端http/https 请求)来判定埋点日志是是否接入正确. 实现细节:使用django项目,后 ...
- IOS-支付宝
一.使用支付宝进行一个完整的支付功能,大致有以下步骤: 与支付宝签约,获得商户ID(partner)和账号ID(seller) 下载相应的公钥私钥文件(加密签名用) 下载支付宝SDK 生成订单信息 ...