题意就不用再说明了吧......如此经典

之前想用双向广搜、a*来写,但总觉得无力,现在用IDA*感觉其他的解法都弱爆了..............想法活跃,时间,空间消耗很小,给它跪了

启发式搜索关键还是找估价函数:此题估价函数可大致定性为每个数字(除去x,只要8个数字)当前位置与它期望位置的曼哈顿距离

即为:v += abs(i - pos[map[i][j] - 1][0]);     v += abs(j - pos[map[i][j] - 1][1]);     大致估算为几十步内得出结果。

#include <iostream>
#include <algorithm>
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <vector>
#include <set>
#include <queue>
#include <stack>
#include <climits>//形如INT_MAX一类的
#define MAX 100005
#define INF 0x7FFFFFFF
#define REP(i,s,t) for(int i=(s);i<=(t);++i)
#define ll long long
#define mem(a,b) memset(a,b,sizeof(a))
#define mp(a,b) make_pair(a,b)
#define L(x) x<<1
#define R(x) x<<1|1
# define eps 1e-5
//#pragma comment(linker, "/STACK:36777216") ///传说中的外挂
using namespace std; int pos[9][2] = { //各个数字的初始位置
{0,0},{0,1},{0,2},
{1,0},{1,1},{1,2},
{2,0},{2,1},{2,2}
};
int map[4][4];
int buff[50];
char input[11];
int limit,ok;
int dx[] = {-1,0,1,0}; //u r d l____0 1 2 3
int dy[] = {0,1,0,-1};
char op[] = {'u','r','d','l'}; int h(int x,int y) {
int v = 0;
for(int i=0; i<3; i++) {
for(int j=0; j<3; j++) {
if(i != x || j != y) {
v += abs(i - pos[map[i][j] - 1][0]);
v += abs(j - pos[map[i][j] - 1][1]);
}
}
}
return v;
} int dfs(int x,int y,int step,int pre) {
int hn = h(x,y);
if(hn == 0) {
ok = 1;
return step;
}
if(hn + step > limit) return hn + step;
int minn = INF;
for(int i=0; i<4; i++) {
if(abs(i - pre) == 2) continue;
int xx = x + dx[i];
int yy = y + dy[i];
if(xx<0 || xx >=3 || yy<0 || yy >=3) continue;
buff[step] = i;
swap(map[x][y],map[xx][yy]);
int tmp = dfs(xx,yy,step+1,i);
if(ok) return tmp;
minn = min(tmp,minn);
swap(map[x][y],map[xx][yy]);
}
return minn;
} void IDA_star(int x,int y) {
ok = 0;
memset(buff,-1,sizeof(buff));
while(ok == 0 && limit <= 36) {
limit = dfs(x,y,0,-111);
}
if(ok == 1) {
for(int i=0; i<limit; i++) printf("%c",op[buff[i]]);
puts("");
} else puts("unsolvable");
} int main() {
for(int i=0; i<9; i++) cin >> input[i];
int t = 0,x,y;
for(int i=0; i<3; i++) {
for(int j=0; j<3; j++) {
if(input[t] == 'x') {
map[i][j] = 9;
x = i;
y = j;
t++;
} else map[i][j] = input[t++] - '0';
}
}
int limit = h(x,y);
if(limit == 0) {
puts("");
return 0;
}
IDA_star(x,y);
return 0;
}

想吐槽一下杭电的这题.........估计各种无法到达目标的数据,所以在输入时候通过求逆序数对数来判断是否有解,本来是TLE,一下蹦到171ms

#include <iostream>
#include <algorithm>
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <string>
#include <vector>
#include <set>
#include <queue>
#include <stack>
#include <climits>//形如INT_MAX一类的
#define MAX 100005
#define INF 0x7FFFFFFF
#define REP(i,s,t) for(int i=(s);i<=(t);++i)
#define ll long long
#define mem(a,b) memset(a,b,sizeof(a))
#define mp(a,b) make_pair(a,b)
#define L(x) x<<1
#define R(x) x<<1|1
# define eps 1e-5
//#pragma comment(linker, "/STACK:36777216") ///传说中的外挂
using namespace std; int pos[9][2] = { //各个数字的初始位置
{0,0},{0,1},{0,2},
{1,0},{1,1},{1,2},
{2,0},{2,1},{2,2}
};
int map[4][4];
int buff[50];
char input[11];
int limit,ok;
int dx[] = {-1,0,1,0}; //u r d l____0 1 2 3
int dy[] = {0,1,0,-1};
char op[] = {'u','r','d','l'}; int h(int x,int y) {
int v = 0;
for(int i=0; i<3; i++) {
for(int j=0; j<3; j++) {
if(i != x || j != y) {
v += abs(i - pos[map[i][j] - 1][0]);
v += abs(j - pos[map[i][j] - 1][1]);
}
}
}
return v;
} int dfs(int x,int y,int step,int pre) {
int hn = h(x,y);
if(hn == 0) {
ok = 1;
return step;
}
if(hn + step > limit) return hn + step;
int minn = INF;
for(int i=0; i<4; i++) {
if(abs(i - pre) == 2) continue;
int xx = x + dx[i];
int yy = y + dy[i];
if(xx<0 || xx >=3 || yy<0 || yy >=3) continue;
buff[step] = i;
swap(map[x][y],map[xx][yy]);
int tmp = dfs(xx,yy,step+1,i);
if(ok) return tmp;
minn = min(tmp,minn);
swap(map[x][y],map[xx][yy]);
}
return minn;
}
int canget(int a[4][4]) { //这一步省了好多时间 求逆序数对数
int i,j,sum=0,b[10],k=0;
for(i=0; i<3; i++) {
for(j=0; j<3; j++) {
if(a[i][j] != 9)
b[++k]=a[i][j];
}
}
for(i=1; i<=8; i++) {
for(j=1; j<i; j++) {
if(b[i]<b[j])sum++;
}
}
if(sum % 2 ==0)return 1;
else return 0;
}
void IDA_star(int x,int y) {
ok = 0;
memset(buff,-1,sizeof(buff));
while(ok == 0 && limit <= 30) {
limit = dfs(x,y,0,-111);
}
if(ok == 1) {
for(int i=0; i<limit; i++) printf("%c",op[buff[i]]);
puts("");
} else puts("unsolvable");
} int main() {
while(cin >> input[0]) {
//memset(map,0,sizeof(map));
for(int i=1; i<9; i++) cin >> input[i];
int t = 0,x,y;
for(int i=0; i<3; i++) {
for(int j=0; j<3; j++) {
if(input[t] == 'x') {
map[i][j] = 9;
x = i;
y = j;
t++;
} else map[i][j] = input[t++] - '0';
}
}
limit = h(x,y);
if(limit == 0) {
puts("");
continue;
}
if(canget(map))
IDA_star(x,y);
else puts("unsolvable");
}
return 0;
}

经过猥琐的测试,limit最小限制在29步....................

POJ 1077 HDU 1043 Eight (IDA*)的更多相关文章

  1. Eight POJ - 1077 HDU - 1043 八数码

    Eight POJ - 1077 HDU - 1043 八数码问题.用hash(康托展开)判重 bfs(TLE) #include<cstdio> #include<iostream ...

  2. POJ 1077 && HDU 1043 Eight A*算法,bfs,康托展开,hash 难度:3

    http://poj.org/problem?id=1077 http://acm.hdu.edu.cn/showproblem.php?pid=1043 X=a[n]*(n-1)!+a[n-1]*( ...

  3. BFS(八数码) POJ 1077 || HDOJ 1043 Eight

    题目传送门1 2 题意:从无序到有序移动的方案,即最后成1 2 3 4 5 6 7 8 0 分析:八数码经典问题.POJ是一次,HDOJ是多次.因为康托展开还不会,也写不了什么,HDOJ需要从最后的状 ...

  4. HDU 1043 & POJ 1077 Eight(康托展开+BFS | IDA*)

    Eight Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 30176   Accepted: 13119   Special ...

  5. HDU 1043 & POJ 1077 Eight(康托展开+BFS+预处理)

    Eight Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 30176   Accepted: 13119   Special ...

  6. HDU - 1043 - Eight / POJ - 1077 - Eight

    先上题目: Eight Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Tota ...

  7. HDU 1403 Eight&POJ 1077(康拖,A* ,BFS,双广)

    Eight Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submis ...

  8. POJ 2104&HDU 2665 Kth number(主席树入门+离散化)

    K-th Number Time Limit: 20000MS   Memory Limit: 65536K Total Submissions: 50247   Accepted: 17101 Ca ...

  9. POJ-1077 HDU 1043 HDU 3567 Eight (BFS预处理+康拓展开)

    思路: 这三个题是一个比一个令人纠结呀. POJ-1077 爆搜可以过,94ms,注意不能用map就是了. #include<iostream> #include<stack> ...

随机推荐

  1. linux系统下安全管理

    1.引导程序安全 linux系统的root密码是很容易破解的,当然前提是你没有设置引导程序密码,如GRUB或LILO,为了防止通过引导程序破译root密码,强烈建 议设置GRUB或LILO的引导密码, ...

  2. Android 为应用添加数字角标

    今天在论坛上看到了一个帖子,终于搞清了我很久以来的一个困惑,android到底能不能实现ios的角标效果,QQ是怎么实现的.看了这个帖子顿时终于解除了我的困惑. 先说一个下大概的思路: 大家都知道an ...

  3. structs常用的Action

    今天座右铭-----谦虚使人进步,骄傲使人落后. 除了基本的Action之外,structs还提供几个其他的类型Action,下面就简单的说一下: 1.DispatchAction:能同时完成多个Ac ...

  4. getline(cin,s) bug workaround

    #include<iostream>using namespace std;#include<string> int main(){int n;stirng s;cin> ...

  5. vim简单命令教程-firstblood

    你想以最快的速度学习人类史上最好的文本编辑器VIM吗?你先得懂得如何在VIM幸存下来,然后一点一点地学习各种戏法. Vim the Six Billion Dollar editor Better, ...

  6. HTTP的头部

    if($this->GetHead("http-edition")=="HTTP/1.1") $httpv = "HTTP/1.1"; ...

  7. 为何与0xff进行与运算

    为何与0xff进行与运算 在剖析该问题前请看如下代码 public static String bytes2HexString(byte[] b) { String ret = "" ...

  8. [转载]解析WINDOWS中的DLL文件---经典DLL解读

    [转载]解析WINDOWS中的DLL文件---经典DLL解读 在Windows世界中,有无数块活动的大陆,它们都有一个共同的名字——动态链接库.现在就走进这些神奇的活动大陆,找出它们隐藏已久的秘密吧! ...

  9. C# 学习笔记 C#基础

    今天第一天开通博客.恰好在学习C#,所以就准备把学到的知识要点记录下来. 基础类型 类型定义了值得蓝图.值是一个被变量或者常量所指定的存储位置,变量是指可以被改变的,而常量则相反,其值不可以便改变, ...

  10. ABCpdf.NET中Rect,Bottom,Height的关系

    因为项目需要新功能,要在一个图片的上下加上固定高度的白边如下图.项目中一直使用ABCpdf.NET处理图片,但我一直没有做这方面的功能,所以找来API参考做. 这个简单需求做的过程中出现了一些曲折,主 ...