[luogu]P1379 八数码难题[广度优先搜索]
八数码难题
——!x^n+y^n=z^n
我在此只说明此题的一种用BFS的方法,因为本人也是初学,勉勉强强写了一个单向的BFS,据说最快的是IDA*(然而蒟蒻我不会…)
各位如果想用IDA*的可以看看这位大佬的这篇文章:
http://www.cnblogs.com/ZYBGMZL/p/6852733.html
接下来是我的方法,用luogu的跑了最慢是200ms,感觉还行把。
题目描述
在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字。棋盘中留有一个空格,空格用0来表示。空格周围的棋子可以移到空格中。要求解的问题是:给出一种初始布局(初始状态)和目标布局(为了使题目简单,设目标状态为123804765),找到一种最少步骤的移动方法,实现从初始布局到目标布局的转变。
输入输出格式
输入格式:
输入初试状态,一行九个数字,空格用0表示
输出格式:
只有一行,该行只有一个数字,表示从初始状态到目标状态需要的最少移动次数(测试数据中无特殊无法到达目标状态数据)
输入输出样例
输入样例#1:
283104765
输出样例#1:
4
因为这是要最优解且保证数据有解,于是就想到了BFS。
然而这个过程是有许多障碍的,要怎样检验自己的状态是否为解,还有判重的操作,如果你没有判重,TLE即在眼前…
所以我们可以想到压缩状态!当然如果用二进制难免有点力不从心,那我们干脆存成整数不就行了?但是可能你会发现,会有前导零的情况,怎么办?
这时候其实可以在状态前加一个1,在int型中还是过得去的。
那判重怎么搞?注意到这只有9!种状态。
想到什么?康托尔展开!对于0~8的全排列,
012345678 的字典序是1,如果让你手动操作我想没什么问题,那怎么让计算机做这件事?
对于 (a(n-1) a(n-2)L a(0))的字典序计算方法为:
Σ(ci*i!)ci为当前未出现的比ai小的数的个数。
既然这样,我们就能把状态一一存下来了,交换的话很简单,读者手动操作即可发现规律。
简单说明之后附上代码参考一下:
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
#define maxn 400000
//阶乘?我当然打表啦。
]={,,,,,,,,,};
//状态
struct sjs{
int num;
int pos;
}state[maxn];
//队列
struct _757{
int time;
int now;
int fd;
}qu[maxn];
];
//判重
bool rep[maxn];
int head,tail;
//特殊嗜好???
namespace lys{
//快速幂
int fpow(int p){
,;
){
) res*=base;
base*=base;
p>>=;
}
return res;
}
//计算cantor
int cantor(int num){
,pos,res=;
int x=num;
memset(cto,,sizeof cto);
){
cto[i]=num%;
) pos=i;
num/=;
i++;
}
int j,cal;
;i>=;i--){
cal=;
;j>i;j--){
if(cto[j]<cto[i]) cal++;
}
res+=(cto[i]-cal)*fac[i];
}
state[res+].num=x;
state[res+].pos=pos;
;
}
//判断是否能移动
bool chk(int pos,int i){
switch(i){
:) return true ; return false ;
:) return true ; return false ;
:||pos==||pos==) return false ; return true ;
:||pos==||pos==) return false ; return true ;
}
}
int bfs(){
,i,st,ch,fp,x,num;
do{
st=qu[head].now;
fp=fpow(state[st].pos);
num=state[st].num;
;i<=;i++){
if(chk(state[st].pos,i)){
switch(i){
:
x=(num/(fp*))%;
ch=num-x*fp*+x*fp;
x=cantor(ch);
qu[++tail].now=x;
qu[tail].time=(qu[head].time+);
qu[tail].fd=head;
//目标态,觉得不这样写也行,直接用num比较
){
return qu[tail].time;
}
if(rep[x]) tail--;
else rep[x]=true ;
break ;
:
x=(num/(fp/))%;
ch=num-x*fp/+x*fp;
x=cantor(ch);
qu[++tail].now=x;
qu[tail].time=(qu[head].time+);
qu[tail].fd=head;
){
return qu[tail].time;
}
if(rep[x]) tail--;
else rep[x]=true ;
break ;
:
x=(num/(fp*))%;
ch=num+x*fp-x*fp*;
x=cantor(ch);
qu[++tail].now=x;
qu[tail].time=(qu[head].time+);
qu[tail].fd=head;
){
return qu[tail].time;
}
if(rep[x]) tail--;
else rep[x]=true ;
break ;
:
x=(num/(fp/))%;
ch=num+x*fp-x*fp/;
x=cantor(ch);
qu[++tail].now=x;
qu[tail].time=(qu[head].time+);
qu[tail].fd=head;
){
return qu[tail].time;
}
if(rep[x]) tail--;
else rep[x]=true ;
break ;
}
}
}
head++;
}while(head<=tail);
}
int main(){
int i,j;
char c;
;
;i<=;i++){
;j<=;j++){
c=getchar();
') c=getchar();
st=st*+c-';
}
}
//初始状态
+st);
qu[++head].now=r;
qu[head].time=;
rep[r]=true ;
tail=;
printf("%d\n",bfs());
;
}
}
int main(){
lys::main();
;
}
[luogu]P1379 八数码难题[广度优先搜索]的更多相关文章
- luogu P1379 八数码难题
题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给出一种初始布局(初始状态)和目标布局(为了 ...
- luogu P1379 八数码难题(A*算法入门详细讲解)
代码实现细节 #include<cstdio> #include<cstring> #include<iostream> using namespace std; ...
- 洛谷 P1379 八数码难题 解题报告
P1379 八数码难题 题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给出一种初始布局(初 ...
- 洛谷——P1379 八数码难题
P1379 八数码难题 双向BFS 原来双向BFS是这样的:终止状态与起始状态同时入队,进行搜索,只不过状态标记不一样而已,本题状态使用map来存储 #include<iostream> ...
- 洛谷P1379八数码难题
题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中. 要求解的问题是:给出一种初始布局(初始状态)和目标布局(为 ...
- codevs1225八数码难题(搜索·)
1225 八数码难题 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 题目描述 Description Yours和zero在研究A*启 ...
- 洛谷—— P1379 八数码难题
https://daniu.luogu.org/problem/show?pid=1379 题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示 ...
- 洛谷 P1379 八数码难题
题目描述 在3×3的棋盘上,摆有八个棋子,每个棋子上标有1至8的某一数字.棋盘中留有一个空格,空格用0来表示.空格周围的棋子可以移到空格中.要求解的问题是:给出一种初始布局(初始状态)和目标布局(为了 ...
- 洛谷 - P1379 - 八数码难题 - bfs
https://www.luogu.org/problemnew/show/P1379 #include <bits/stdc++.h> using namespace std; #def ...
随机推荐
- OpenCV中的结构体、类与Emgu.CV的对应表
OpenCv中的 C 结构 OpenCV中的 C++ 封装 Emgu.CV中的 C# 封装 OpenCV 和 Emgu.CV 中的结构罗列 谢谢阅读,有误希望指正 原文地址 Basic Structu ...
- “海市蜃楼”般的逛街体验——VR全景智慧城市常诚
<史记·天官书>:"海旁蜃气像楼台:广野气成宫阙然." 海市蜃楼,简称蜃景,是一种因为光的折射和全反射而形成的自然现象,是地球上物体反射的光经大气折射而形成的虚像. 2 ...
- 在.net下打造mongoDb基于官方驱动最新版本
还是一如既往先把结构图放出来,上上个版本添加了redis的缓存,但是不满足我的需求,因为公司有项目要求是分布式所以呢,这里我就增加了mongoDb进行缓存分布式,好了先看结构图. 总的来说比较蛋疼,因 ...
- javascript的八张图
- 美丽的CSS图形和HTML5
1.HTML5基础 文档类型 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "htt ...
- 浅析TCP/IP 协议
TCP/IP协议不是TCP和IP这两个协议的合称,而是指因特网整个TCP/IP协议族. TCP/IP协议模块关系 从协议分层模型方面来讲,TCP/IP由四个层次组成:网络接口层.网络层.传输层.应用层 ...
- Google云平台技术架构
Google Cloud 设计原理: 1.分布式文件系统: Google Distributed File System(GSF) 为了满足Google迅速增长的数据处理需求,我们设计并实现了G ...
- Redis中的基本数据结构
Redis基础数据结构 基础数据结构 sds简单动态字符串 数据结构 typedef struct sdstr{ int len // 字符串分配的字节 int free // 未使用的字节数 cha ...
- google和oracle闹掰,Java 会不会被抛弃?
眼花缭乱的编程语言 程序界的语言实在太多,但有一种语言不得不说,那就是java语言,Java语言是Android系统的主要开发语言,现在和Google的关系不是很好,但是他会被淘汰吗?下面简单地分析一 ...
- HTML行内元素、块状元素、行内块状元素的区别
HTML可以将元素分类方式分为行内元素.块状元素和行内块状元素三种.首先需要说明的是,这三者是可以互相转换的,使用display属性能够将三者任意转换: (1)display:inline;转换为行内 ...