双向BFS (广搜) \(O(8 ^ 7)\)


看到没有双向BFS的题解我就过来了

这道题也可以用双向\(BFS\)来做,时间复杂度与\(IDA*\)不相上下。

双向\(BFS\)的实现有多种:

  1. 把初始状态和目标状态扔在一个队列里,每次从队列里搞出来一个扩展
  2. 把初始状态和目标状态扔在两个队列里,每次选一个队列中元素少的拓展。
  3. 把初始状态和目标状态扔在两个队列里,每次分别从两个队列中取出一个元素拓展。

这里使用了方法\(1\)。


时间复杂度分析:每次会扩展\(8\)个状态,最多扩展\(\lfloor\frac{15}{2}\rfloor = 7\)次(到第八次如果还找不到则肯定超过限制的\(15\)步,答案则无需扩展),所以复杂度是\(O(8 ^ 7)\)的

具体实现方面看代码。

C++ 代码

#include <iostream>
#include <cstdio>
#include <unordered_map>
#include <queue>
using namespace std;
typedef pair<int, int> PII;
const int INF = 0x3f3f3f3f;
int dx[8] = {1, 1, -1, -1, 2, 2, -2, -2};
int dy[8] = {-2, 2, 2, -2, 1, -1, 1, -1};
//移动方向的打表
int n;
//状态结构体
struct Node{
int v[5][5], f, x, y;
//f = 0 表示从初始状态出发, f = 1 表示从目标状态出发
}s, t, u;
unordered_map<int, int> step[2][30];
// 记录每个状态对应的位置。
int goal[5][5] = {
{1, 1, 1, 1, 1},
{0, 1, 1, 1, 1},
{0, 0, 2, 1, 1},
{0, 0, 0, 0, 1},
{0, 0, 0, 0, 0}
};
//目标状态打表
int start[5][5]; //get(x) 获取状态x的步数
int get(Node x){
int sum = 0;
//把一个位置状态转移成一个二进制数,比较方便。
for(int i = 0; i < 5; i++)
for(int j = 0; j < 5; j++)
if(x.v[i][j] == 1)
sum |= 1 << (i * 5 + j);
if(step[x.f][x.x * 5 + x.y].count(sum) == 0) return INF;
return step[x.f][x.x * 5 + x.y][sum];
}
//set() 设置状态x的步数
void set(Node x, int val){
int sum = 0;
for(int i = 0; i < 5; i++)
for(int j = 0; j < 5; j++)
if(x.v[i][j] == 1)
sum |= 1 << (i * 5 + j);
step[x.f][x.x * 5 + x.y][sum] = val;
}
int bfs(){
queue<Node> q;
q.push(s); q.push(t);
set(s, 0);
set(t, 0);
while(!q.empty()){
Node u = q.front(); q.pop(); //找到同位置的另一状态
Node l = u; l.f = u.f ^ 1; //如果相遇,代表已经找到一条最短的路径
if(get(u) + get(l) <= 15)
return get(u) + get(l); //说明所有状态已经都至少走了 8 步, > 最大限制 15
if(get(u) >= 8) break; l.f = u.f;
for(int i = 0; i < 8; i++){
int nx = u.x + dx[i], ny = u.y + dy[i];
if(nx < 0 || nx >= 5 || ny < 0 || ny >= 5) continue;
swap(l.v[u.x][u.y], l.v[nx][ny]);
l.x = nx, l.y = ny; if(get(u) + 1 < get(l)){
set(l, get(u) + 1);
q.push(l);
}
//还原现场
swap(l.v[u.x][u.y], l.v[nx][ny]);
}
}
return -1;
}
int main(){
for(int i = 0; i < 5; i++)
for(int j = 0; j < 5; j++)
t.v[i][j] = goal[i][j];
t.f = 1; t.x = 2; t.y = 2;
int T; scanf("%d", &T);
while(T--){
for(int i = 0; i < 2; i++)
for(int j = 0; j < 30; j++)
step[i][j].clear();
for(int i = 0; i < 5; i++){
for(int j = 0; j < 5; j++){
char x; cin >> x;
if(x == '*')s.x = i, s.y = j, s.v[i][j] = 2;
else s.v[i][j] = x - '0';
}
}
s.f = 0;
printf("%d\n", bfs());
}
return 0;
}

AcWing 195. 骑士精神的更多相关文章

  1. BZOJ1085: [SCOI2005]骑士精神 [迭代加深搜索 IDA*]

    1085: [SCOI2005]骑士精神 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1800  Solved: 984[Submit][Statu ...

  2. A*算法详解 BZOJ 1085骑士精神

    转载1:A*算法入门 http://www.cppblog.com/mythit/archive/2009/04/19/80492.aspx 在看下面这篇文章之前,先介绍几个理论知识,有助于理解A*算 ...

  3. 【BZOJ】1085: [SCOI2005]骑士精神(A*启发式搜索)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1085 囧啊囧,看了题解后写了个程序,但是样例总过不了T+T,调试了不下于1个小时,肉眼对拍看了根本看 ...

  4. 【bzoj1085】[SCOI2005]骑士精神

    1085: [SCOI2005]骑士精神 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1757  Solved: 961[Submit][Statu ...

  5. 【BZOJ 1085】 [SCOI2005]骑士精神

    Description 在一个5×5的棋盘上有12个白色的骑士和12个黑色的骑士, 且有一个空位.在任何时候一个骑士都能按照骑士的走法(它可以走到和它横坐标相差为1,纵坐标相差为2或者横坐标相差为2, ...

  6. Codevs 2449 骑士精神 2005年省队选拔赛四川

    2449 骑士精神 2005年省队选拔赛四川 时间限制: 1 s 空间限制: 128000 KB 题目等级 : **大师 Master** 题目描述 Description 在一个5×5的棋盘上有12 ...

  7. BZOJ 1085: [SCOI2005]骑士精神( IDDFS + A* )

    一开始写了个 BFS 然后就 T 了... 这道题是迭代加深搜索 + A* -------------------------------------------------------------- ...

  8. [bzoj1805][SCOI2005]骑士精神 [启发式搜索]

    Description 在一个5×5的棋盘上有12个白色的骑士和12个黑色的骑士, 且有一个空位.在任何时候一个骑士都能按照骑士的走法(它可以走到和它横坐标相差为1,纵坐标相差为2或者横坐标相差为2, ...

  9. 【日常学习】【IDA*】codevs2449 骑士精神题解

    题目描写叙述 Description 在一个5×5的棋盘上有12个白色的骑士和12个黑色的骑士, 且有一个空位.在不论什么时候一个骑士都能依照骑士的走法(它能够走到和它横坐标相差为1.纵坐标相差为2或 ...

随机推荐

  1. mysql建立索引,实际工作中建立索引的示例

    1.根据业务场景建立相应的组合索引,一般是在主键,外键,常用来筛选查询的字段,按照字段之间组合的紧密程度,建立一定顺序的索引. 例如:为 t_org_exam_join_member_day  建立索 ...

  2. 【开发实录】在鸿蒙开发板上使用websocket(移植自librws库)

    librws: Tiny, cross platform websocket client C library 相关代码可在下面下载,也可进入librws: 将librws移植到鸿蒙Hi_3861开发 ...

  3. 《Machine Learning in Action》—— 小朋友,快来玩啊,决策树呦

    <Machine Learning in Action>-- 小朋友,快来玩啊,决策树呦 在上篇文章中,<Machine Learning in Action>-- Taoye ...

  4. vue项目中h5移动端中通过flex布局实现首尾固定,中间滚动(借鉴)

    html中 <div class="flexLayoutr"> <div class="div_head"></div> & ...

  5. 使用python统计《三国演义》小说里人物出现次数前十名,并实现可视化。

    一.安装所需要的第三方库 jieba (jieba是优秀的中文分词第三分库) pyecharts (一个优秀的数据可视化库) <三国演义>.txt下载地址(提取码:kist ) 使用pyc ...

  6. 直面秋招!非科班生背水一战,最终拿下阿里等大厂offer!

    前言 2020年已经接近到9月份了,很多粉丝朋友都对金九银十雀雀欲试了吧!也有很多朋友向我求教经验,因为我自己工作相对于稳定,在这里给大家分享一个粉丝朋友的经历,他作为一个曾经的菜鸡面试者,在不断的失 ...

  7. 带你体验Folx的智能化下载功能

    BT下载器,也就是种子下载器,其优点是下载的人越多,下载的速度就越快,可以更快速地下载热门资源.这是因为,每个下载种子的用户,在下载的同时,也会上传资源,从而加速其他用户的下载速度. Folx软件,作 ...

  8. 用Camtasia来快速地给视频添加水印

    在日常生活中,视频的流行度越来越高,各种短视频的软件蜂拥上市,所以越来越多的人走上了自媒体的道路,在这条路上,谁的视频更加的精致,谁才能获得更多的关注度,相应的也能增加自己的人气. 但是在制作视频的过 ...

  9. Tuxera Disk Manager轻松解决硬盘格式转换问题

    生活中经常会遇到硬盘格式转换的问题,很多小伙伴都不知道怎么进行操作,特别是Mac小白们.今天,小编想要给不熟悉Mac系统的小伙伴推荐一款专业且高效的磁盘管理工具--Tuxera NTFS,可以帮助我们 ...

  10. thinkPHP 无法加载控制器:Hello

    出现这种问题的情况下要看看: (1).控制器的名称是否写对,控制器的命名规范(别忘下class) 控制器的命名规则1.必须采用大驼峰的命名规则2.必须以Controller.class.php来结尾I ...