这题我第一次用的bfs + ELFhash,直接TLE,又换成bfs + 康托还是TLE,5000ms都过不了!!我一直调试,还是TLE,我才发觉应该是方法的问题。

今天早上起床怒学了一波A*算法,因为IDA*我很熟悉,因此A*也学得很快,曼哈顿距离也很容易理解,看了好多人都用的A*过掉的。我一直在想A*算法无法保证得到最短路啊,怎么能AC?我擦,后来一读题目,题目没说要最短路径啊,只要任意一条路就可以了。我就呵呵了,愤怒写了一波A*过掉。

思路:

1. 根据逆序数可以判断是否有解,如果当前逆序数为奇数一定无解,是偶数则进行搜索。

2. 利用康拓判重。

3.  启发函数 : F = G + H. A*算法的精髓, G表示从初始状态到达当前状态的花费,H表示当前状态到达目标状态的花费,H只能大概估计,估计可用曼哈顿距离。曼哈顿距离介绍

AC代码

#include<cstdio>
#include<queue>
#include<cmath>
#include<cstring>
#include<algorithm>
using namespace std;

const int maxn = 4e5 + 5;
int vis[maxn]; //close表用于判重
int goal[] = {1,2,3,4,5,6,7,8,0};
int st[9];
typedef int state[9];

struct node{
	int a[9];
	int g, h; // 估价函数
	int ha, pos;
	node() {
	}
	node(int *b, int g, int h, int ha, int pos):g(g), h(h), ha(ha), pos(pos){
		memcpy(a, b, sizeof(a));
	}
	bool operator < (const node &p) const {
		return h > p.h || (p.h == h && g > p.g); // h小在前,g小在前
	}
};

struct Node{
	int pre;
	char dir;
}Pre[maxn];

const int dx[] = {0,0,-1,1};
const int dy[] = {1,-1,0,0};
const char dir[] = {'r','l','u','d'};

int fact[9];
void deal() {  //1~8阶乘打表,方便编码
	fact[0] = 1;
	for(int i = 1; i < 9; ++i) fact[i] = fact[i - 1] * i;
}

int cut(int *a) {
	int cnt = 0;
	for(int i = 0; i < 9; ++i) {
		if(a[i] == 0) continue;
		for(int j = i + 1; j < 9; ++j) {
			if(a[j] != 0 && a[i] < a[j]) ++cnt;
		}
	}
	return cnt;
} 

int KT(int *a) {
	int code = 0;
	for(int i = 0; i < 9; ++i) {
		int cnt = 0;
		for(int j = i + 1; j < 9; ++j) if(a[j] < a[i]) cnt++;
		code += fact[8 - i] * cnt;
	}
	return code;
} 

void print(int u) {
	if(Pre[u].pre == -1) return;
	print(Pre[u].pre);
	printf("%c", Pre[u].dir);
}

int get_h(int *a) { //根据曼哈顿距离得到的估价函数
	int cnt = 0;
	for(int i = 0; i < 9; ++i) {
		if(!a[i]) continue;
		int x = (a[i] - 1) / 3, y = (a[i] - 1) % 3; //终点的position
		cnt += abs(i / 3 - x) + abs(i % 3 - y);
	}
	return cnt;
}

int Astar() {
	priority_queue<node>q;
	int des = KT(goal); //目标
	memset(vis, 0, sizeof(vis));
	int code = KT(st);
	vis[code] = 1;
	Pre[code].dir = 'x';
	Pre[code].pre = -1;
	int pos;
	for(int i = 0; i < 9; ++i) if(!st[i]) pos = i;
	int g = 0, h = get_h(st);
	q.push(node(st, g, h, code, pos));
	while(!q.empty()) {
		node p = q.top();
		q.pop();
		code = p.ha, g = p.g, h = p.h, pos = p.pos;
		if(code == des) {  // 找到答案
			//printf("%d\n", g);
			print(des);
			return 1;
		}
		state &a = p.a;
		int x = pos / 3, y = pos % 3;
		for(int i = 0; i < 4; ++i) {
			int px = x + dx[i], py = y + dy[i];
			if(px < 0 || py < 0 || px >= 3 || py >= 3) continue;
			int pz = px * 3 + py;
			swap(a[pos], a[pz]);
			int code1 = KT(a);
			if(vis[code1]) {
				swap(a[pos], a[pz]);
				continue;
			}
			vis[code1] = 1;
			Pre[code1].dir = dir[i];
			Pre[code1].pre = code;
			q.push(node(a, g + 1, get_h(a), code1, pz));
			swap(a[pos], a[pz]);
		}
	}
	return -1;
}

int main() {
	deal();
	char str[50];
	while(fgets(str, sizeof(str), stdin) != NULL) {
		int ind = 0;
		for(int i = 0; str[i] != '\n'; ++i) {
 			if(str[i] == 'x') st[ind++] = 0;
			else if(str[i] >= '0' && str[i] <= '9') st[ind++] = str[i] - '0';
		}
		// 根据逆序数提前判定是否有解
		//for(int i = 0; i < 9; ++i) printf("%d ", st[i]);
		if(cut(st) % 2 == 1) {
			printf("unsolvable\n");
			continue;
		}
		if( Astar() == -1) printf("unsolvable");
		printf("\n");
	}
	return 0;
}

如有不当之处欢迎指出!

HDU - 1043 A* + 康托 [kuangbin带你飞]专题二的更多相关文章

  1. HDU - 3567 Eight II (bfs预处理 + 康托) [kuangbin带你飞]专题二

    类似HDU1430,不过本题需要枚举X的九个位置,分别保存状态,因为要保证最少步数.要保证字典序最小的话,在扩展节点时,方向顺序为:down, left, right, up. 我用c++提交1500 ...

  2. HDU - 3567 IDA* + 曼哈顿距离 + 康托 [kuangbin带你飞]专题二

    这题难度颇大啊,TLE一天了,测试数据组数太多了.双向广度优先搜索不能得到字典序最小的,一直WA. 思路:利用IDA*算法,当前状态到达目标状态的可能最小步数就是曼哈顿距离,用于搜索中的剪枝.下次搜索 ...

  3. 【算法系列学习三】[kuangbin带你飞]专题二 搜索进阶 之 A-Eight 反向bfs打表和康拓展开

    [kuangbin带你飞]专题二 搜索进阶 之 A-Eight 这是一道经典的八数码问题.首先,简单介绍一下八数码问题: 八数码问题也称为九宫问题.在3×3的棋盘,摆有八个棋子,每个棋子上标有1至8的 ...

  4. HDU - 3001 Travelling 状压dp + 三进制 [kuangbin带你飞]专题二

    终于刷完搜索专题了. 题意:给定n个城市,每个城市参观不能超过两次,两个城市之间有道路通过需要花费X,求通过能所有城市的最小花费. 思路:每个城市有三个状态0,1,2,可用三进制存储所有城市的访问状态 ...

  5. HDU - 3533 bfs [kuangbin带你飞]专题二

    看了好久的样例才看懂. 题意:有一个人要从(0,0)走到(n,m),图中有k个碉堡,每个碉堡可以向某个固定的方向每隔t秒放一次炮,炮弹不能穿越另一个碉堡,会被阻挡.人在移动的过程中不会被炮弹打到,也就 ...

  6. HDU - 3085 双向BFS + 技巧处理 [kuangbin带你飞]专题二

    题意:有两只鬼,一个男孩女孩被困在迷宫中,男孩每秒可以走三步,女孩只能1步,鬼可以两步且可以通过墙.问男孩女孩是否可以在鬼抓住他们之前会合? 注意:每秒开始鬼先移动,然后两人开始移动. 思路:以男孩和 ...

  7. HDU - 1067 Gap (bfs + hash) [kuangbin带你飞]专题二

    题意:    起初定28张卡牌的排列,把其中11,  21, 31, 41移动到第一列,然后就出现四个空白,每个空白可以用它的前面一个数的下一个数填充,例如43后面的空格可以用44填充,但是47后面即 ...

  8. HDU - 2102 A计划 (BFS) [kuangbin带你飞]专题二

    思路:接BFS判断能否在限制时间内到达公主的位置,注意如果骑士进入传送机就会被立即传送到另一层,不会能再向四周移动了,例如第一层的位置(x, y, 1)是传送机,第二层(x, y, 2)也是传送机,这 ...

  9. HDU - 2181 dfs [kuangbin带你飞]专题二

    保存每个节点的下一个节点一直往下面走就行了,不能重复经过某个点,当经过的点达到20个而且当前节点的下一个节点是起点就打印答案. AC代码 #include<cstdio> #include ...

随机推荐

  1. JS动态获取当前时间

    HTML部分: <div class="div"> <div id="div"> </div> </div> C ...

  2. ImmutableMap

    不可变集合,为什么使用它呢?线程安全\更有效的利用内存\可作为常量 ImmutableMap.<String, Object> builder().put("yananList& ...

  3. 【Spring实战】--1Spring的核心

    最近面试总会涉及Spring的优点,SpringMVC与Struts2的比较,生活慢慢稳定下来,这些面试还是应了那句话“只顾盲目拉车,不会低头看路”,回过头来还是要好好研究一下Spring,如果仅仅是 ...

  4. js中键盘按键对应的键值

    js键盘键值 keycode    8 = BackSpace BackSpace  keycode    9 = Tab Tab  keycode   12 = Clear  keycode   1 ...

  5. 关于Spring的注入方式

      spring的三种注入方式: 接口注入(不推荐) getter,setter方式注入(比较常用) 构造器注入(死的应用) 关于getter和setter方式的注入: autowire=" ...

  6. Global对象

    1.Global对象是不需要创建实例就可以直接调用方法或属性的对象.有点类是于java中的工具类 2.关于汉字的网络传递 网络访问的过程中在传递数据的时候,传递的本质都是0101,如果你要传送字符的话 ...

  7. 使用redis所维护的代理池抓取微信文章

    搜狗搜索可以直接搜索微信文章,本次就是利用搜狗搜搜出微信文章,获得详细的文章url来得到文章的信息.并把我们感兴趣的内容存入到mongodb中. 因为搜狗搜索微信文章的反爬虫比较强,经常封IP,所以要 ...

  8. 单元测试系列:Mock工具之Mockito实战

    更多原创测试技术文章同步更新到微信公众号 :三国测,敬请扫码关注个人的微信号,感谢! 原文链接:http://www.cnblogs.com/zishi/p/6780719.html 在实际项目中写单 ...

  9. MOBA服务器开发第一阶段完成总结

    开发历程 项目是从8月20日左右开始开发的,到今天一个月不到吧. 除了底层库和服务器架构外我们大致开发了5个服务器为: 一 ) . 战斗服务器 二 ) . 匹配服务器 三 ) . 验证服务器 四 ) ...

  10. POJ1269 Intersecting Lines[线段相交 交点]

    Intersecting Lines Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 15145   Accepted: 66 ...