POJ2308连连看dfs+bfs+优化
DFS+BFS+MAP+剪枝
题意:
就是给你一个10*10的连连看状态,然后问你最后能不能全部消没?
思路:
首先要明确这是一个搜索题目,还有就是关键的一点就是连连看这个游戏是存在决策的,就是如果当前的这个点可以连接好几个点的话,我们选择那个点连接是不一样的,所以还要遍历所有的可能连接点,这样考虑的话单纯的一个搜索肯定不行了,可以用一个深搜套广搜的的结构,深搜是为了遍历顺序,广搜所为了得到当前点可以消去那些点(必要一个一个搜,要直接把所有的可能都搜出来,就是把杭电的那个bfs扩展下),但是直接暴力时间复杂度肯定接受不了,目测是,目测是多少我也不知道,但是存在这样的数据t = O(100^25),所以要优化,关键是在优化,下面是我能想到的几个优化
(1)首先把所有的卡片都单独拿出来,放到一个结构体里面,这样每次遍历的时候直接就只遍历数组,不用遍历所有的图。
(2)直接判断每种卡片出现的次数的奇偶。
(3)还有就是开一个状态记录当前的这个状态是否出现过,这个我是用的map进行hash的,这个优化有种记忆化搜索的感觉。
(4)还有最后一个(百度的),也是最关键的,如果有两种卡片都只剩下两个了,并且出现这样的姿势
AB
BA
显然这样是连接不了的,这样感觉不会优化多少,但是在这个题目里,这样貌似优化很大,原因我感觉是在卡片的种类上的原因,卡片种类只有4种,这个也只是猜测,其实我力推的优化是(2,3)可惜没有4一直超时。
还有就是一定要明确,连连看是存在决策问题的。
#include<map>
#include<queue>
#include<stdio.h>
#include<string.h>
#include<string>
using namespace std;
typedef struct
{
int x ,y ,t;
}NODE;
typedef struct
{
int x ,y;
}CARD;
NODE xin ,tou;
CARD card[105];
CARD can[105];
int ss[5];
int cans ,mkans ,n ,m ,nowc ,cards;
int _map[12][12];
int dir[4][2] = {0 ,1 ,0 ,-1 ,1 ,0 ,-1 ,0};
map<string ,int>MARK;
bool ok(int x ,int y)
{
return x >= 1 && x <= n && y >= 1 && y <= m && (!_map[x][y] || _map[x][y] == nowc);
}
bool jude()
{
char str[105];
for(int i = 1 ;i <= cards ;i ++)
if(_map[card[i].x][card[i].y]) str[i-1] = 'a';
else str[i-1] = 'b';
str[cards] = '\0';
if(MARK[str]) return 0;
MARK[str] = 1;
return 1;
}
int BFS(int x ,int y)
{
int mark[12][12] = {0};
nowc = _map[x][y];
cans = 0;
xin.x = x ,xin.y = y ,xin.t = 0;
queue<NODE>q;
q.push(xin);
while(!q.empty())
{
tou = q.front();
q.pop();
if(_map[tou.x][tou.y] == nowc && !(tou.x == x && tou.y == y))
{
cans ++;
can[cans].x = tou.x;
can[cans].y = tou.y;
continue;
}
if(tou.t >= 3) continue;
for(int i = 0 ;i < 4 ;i ++)
{
xin.x = tou.x ,xin.y = tou.y ,xin.t = tou.t + 1;
while(1)
{
xin.x += dir[i][0];
xin.y += dir[i][1];
if(!ok(xin.x ,xin.y)) break;
if(!mark[xin.x][xin.y])
{
mark[xin.x][xin.y] = 1;
q.push(xin);
}
}
}
}
return cans;
}
bool CCC(int a ,int b)
{
for(int i = 1 ;i <= cards ;i ++)
{
int x = card[i].x ,y = card[i].y;
if(_map[x][y] == a)
{
if(_map[x+1][y+1] == a && _map[x+1][y] == b && _map[x][y+1] == b)
return 1;
return 0;
}
if(_map[x][y] == b)
{
if(_map[x+1][y+1] == b && _map[x+1][y] == a && _map[x][y+1] == a)
return 1;
return 0;
}
}
}
bool CUT()
{
for(int i = 1 ;i <= 4 ;i ++)
for(int j = i + 1 ;j <= 4 ;j ++)
{
if(ss[i] == ss[j] && ss[i] == 2)
{
if(CCC(i ,j)) return 1;
}
}
return 0;
}
void DFS(int nows)
{
if(!nows) mkans = 1;
if(mkans) return ;
if(!jude()) return ;
if(CUT()) return ;
for(int i = 1 ;i <= cards ;i ++)
{
if(mkans) return ;
int x = card[i].x ,y = card[i].y;
if(!_map[x][y]) continue;
BFS(x ,y);
ss[_map[x][y]] -= 2;
for(int j = 1 ;j <= cans ;j ++)
{
int tmp = _map[x][y];
int xx = can[j].x ,yy = can[j].y;
_map[x][y] = _map[xx][yy] = 0;
DFS(nows - 2);
_map[x][y] = _map[xx][yy] = tmp;
}
ss[_map[x][y]] += 2;
}
}
int main ()
{
char str[12];
while(~scanf("%d %d" ,&n ,&m) && n + m)
{
memset(_map ,0 ,sizeof(_map));
memset(ss ,0 ,sizeof(ss));
cards = 0;
for(int i = 1 ;i <= n ;i ++)
{
scanf("%s" ,str);
for(int j = 1 ;j <= m ;j ++)
{
if(str[j-1] == '*') _map[i][j] = 0;
else
{
_map[i][j] = str[j-1] - 'A' + 1;
cards ++;
card[cards].x = i;
card[cards].y = j;
ss[_map[i][j]] ++;
}
}
}
if(ss[1]%2 || ss[2]%2 || ss[3]%2 || ss[4]%2)
{
printf("no\n");
continue;
}
mkans = 0;
MARK.clear();
DFS(cards);
mkans ? printf("yes\n"):printf("no\n");
}
return 0;
}
POJ2308连连看dfs+bfs+优化的更多相关文章
- DFS/BFS+思维 HDOJ 5325 Crazy Bobo
题目传送门 /* 题意:给一个树,节点上有权值,问最多能找出多少个点满足在树上是连通的并且按照权值排序后相邻的点 在树上的路径权值都小于这两个点 DFS/BFS+思维:按照权值的大小,从小的到大的连有 ...
- 【DFS/BFS】NYOJ-58-最少步数(迷宫最短路径问题)
[题目链接:NYOJ-58] 经典的搜索问题,想必这题用广搜的会比较多,所以我首先使的也是广搜,但其实深搜同样也是可以的. 不考虑剪枝的话,两种方法实践消耗相同,但是深搜相比广搜内存低一点. 我想,因 ...
- ID(dfs+bfs)-hdu-4127-Flood-it!
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=4127 题目意思: 给n*n的方格,每个格子有一种颜色(0~5),每次可以选择一种颜色,使得和左上角相 ...
- hdu 4109 dfs+剪枝优化
求最久时间即在无环有向图里求最远路径 dfs+剪枝优化 从0节点(自己添加的)出发,0到1~n个节点之间的距离为1.mt[i]表示从0点到第i个节点眼下所得的最长路径 #include<iost ...
- [LeetCode] 130. Surrounded Regions_Medium tag: DFS/BFS
Given a 2D board containing 'X' and 'O' (the letter O), capture all regions surrounded by 'X'. A reg ...
- HDU 4771 (DFS+BFS)
Problem Description Harry Potter has some precious. For example, his invisible robe, his wand and hi ...
- DFS/BFS视频讲解
视频链接:https://www.bilibili.com/video/av12019553?share_medium=android&share_source=qq&bbid=XZ7 ...
- 2013年第四届蓝桥杯国赛 九宫重排(HashMap+双BFS优化)
九宫重排 时间限制:1.0s 内存限制:256.0MB 问题描述 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干 ...
- POJ 3083 -- Children of the Candy Corn(DFS+BFS)TLE
POJ 3083 -- Children of the Candy Corn(DFS+BFS) 题意: 给定一个迷宫,S是起点,E是终点,#是墙不可走,.可以走 1)先输出左转优先时,从S到E的步数 ...
随机推荐
- [Elementary Mechanics-01]Two masses and a spring
[Elementary Mechanics Using Python-01] Question 5.28 Two masses and a spring. Two particles of m = 0 ...
- Typora学习
Markdown学习总结 标题的使用格式 # 一阶标题 或者 ctrl + 1 ## 二阶标题 或者 ctrl + 2 ### 三阶标题 或者 ctrl + 3 #### 四阶标题 或者 ctrl + ...
- 设计模式之简单工厂模式(Simple Factory Pattern)
一.简单工厂模式的由来 所有设计模式都是为解决某类问题而产生的,那么简单工厂模式是为解决什么问题呢?我们假设有以下业务场景: 在一个学生选课系统中,文科生用户选课时,我们要获得文科生的所有课程列表:理 ...
- Java 哈希表(google 公司的上机题)
1 哈希表(散列)-Google 上机题 1) 看一个实际需求,google 公司的一个上机题: 2) 有一个公司,当有新的员工来报道时,要求将该员工的信息加入(id,性别,年龄,住址..),当输入该 ...
- 基于es实现一个简单的搜索引擎
一.什么是es Elasticsearch是一个基于ApacheLucene(TM)的开源搜索引擎.无论在开源还是专有领域,Lucene可以被认为是迄今为止最先进.性能最好的.功能最全的搜索引擎库.但 ...
- apk签名、包名
//通过各手机管理软件,如如360.豌豆荚等查看 //使用命令行,可以查看到permission.packagename.title.versionCode等 aapt dump badging ~/ ...
- Vue ElementUI表格table中使用select下拉框组件时获取改变之前的值
目前项目中有一个场景,就是表格中显示下拉框,并且下拉框的值可以更改,更改后提交后台更新.因为这个操作比较重要,所以切换时会有一个提示框,提示用户是否修改,是则走提交逻辑,否则直接返回,什么也不做. 之 ...
- 解决unbutu网络编程socket_tcp连接不上网络助手
unbutu开放指定端口 开放端口8080 sudo iptables -I INPUT -p tcp --dport 8080 -j ACCEPT 保存设置 iptables-save 在终端中输入 ...
- UML类图画法整理
一 类图画法 1.类图的概念 显示出类.接口以及他们的静态结构和关系,用于描述系统的结构化设计. 2.类 类是对一组具有相同属性.操作.关系和语义对象的抽象,是面向对象的核心,包括名称.属性和方法.如 ...
- Hashtable 渐渐被人们遗忘了,只有面试官还记得,感动
尽人事,听天命.博主东南大学硕士在读,热爱健身和篮球,乐于分享技术相关的所见所得,关注公众号 @ 飞天小牛肉,第一时间获取文章更新,成长的路上我们一起进步 本文已收录于 「CS-Wiki」Gitee ...