HDU - 1067 Gap (bfs + hash) [kuangbin带你飞]专题二
题意: 起初定28张卡牌的排列,把其中11, 21, 31, 41移动到第一列,然后就出现四个空白,每个空白可以用它的前面一个数的下一个数填充,例如43后面的空格可以用44填充,但是47后面即使有空白也无法填充,因为没有48。
思路:每次有四个空格,面临四个选择,直接用bfs搜索,状态保存的话就自己设计一个哈希函数,如果找不到完美哈希一定要解决冲突。
我设计的哈希函数:我用的是双哈希,第一个哈希值用于查找,第二个哈希值可以看做整个卡牌状态的压缩,如果有两种状态有同一个第一哈希,我就用第二哈希来判断它们是否一致(第二哈希也一样的可能性几乎没有)。我的时间是1201ms,有的200多ms就过了,说明决定这题快不快就在于哈希函数判重快不快。
注意:空白前面可能也是空白哟。
AC代码: 1201ms
#include <iostream> #include <stdio.h> #include <string.h> #include <algorithm> #include <queue> #include<utility> using namespace std; typedef long long LL; typedef pair<int, int> PI; typedef int state[4][8]; const int maxn = 1e6 + 10; const int hash1 = 1000007, hash2 = 1007; //双哈希解决冲突 int goal[][8] = { 11,12,13,14,15,16,17,0, 21,22,23,24,25,26,27,0, 31,32,33,34,35,36,37,0, 41,42,43,44,45,46,47,0 }; int st[4][8]; int sp[][2] = { 0,0, 1,0, 2,0, 3,0 }; struct node{ int pip; int next; node() { } node(int pip, int next): pip(pip), next(next){ } }HASH[maxn]; int head[maxn]; struct card{ int a[4][8]; int pos[4][2]; //四个空位 int step; card() { } card(int b[][8], int p[][2], int step):step(step) { memcpy(a, b, sizeof(a)); memcpy(pos, p, sizeof(pos)); } }; PI get_hash(int a[][8]) { LL h1 = 0, h2 = 0; for(int i = 0; i < 4; ++i) for(int j = 0; j < 8; ++j) { h1 = (h1 * 100 + a[i][j]) % hash1; h2 = (h2 * 100 + a[i][j]) % hash2; } return make_pair((int)h1, (int)h2); } int cur = 0; bool Insert(PI pi) { int h = pi.first, val = pi.second; int u = head[h]; while(u != -1) { if(val == HASH[u].pip) return false; u = HASH[u].next; } HASH[cur] = node(val, head[h]); head[h] = cur; cur++; return true; } int find(int a[][8], int key) { for(int i = 0; i < 4; ++i) for(int j = 0; j < 8; ++j) { if(a[i][j] == key) return i * 8 + j; } } int bfs() { memset(head, -1, sizeof(head)); queue<card>q; q.push(card(st, sp, 0)); PI pi = get_hash(st); Insert(pi); while(!q.empty()) { card cd = q.front(); q.pop(); state &a = cd.a; if(!memcmp(goal, a, sizeof(goal))) return cd.step; for(int i = 0; i < 4; ++i) { int x = cd.pos[i][0], y = cd.pos[i][1]; int val = a[x][y - 1]; if(val == 0 || val % 10 == 7) continue; int pos = find(a, val + 1); int px = pos / 8, py = pos % 8; swap(a[x][y], a[px][py]); PI pi = get_hash(a); if(!Insert(pi)) { //判重 swap(cd.a[x][y], cd.a[px][py]); continue; } cd.pos[i][0] = px, cd.pos[i][1] = py; cd.step++; q.push(cd); cd.pos[i][0] = x, cd.pos[i][1] = y; swap(a[x][y], a[px][py]); cd.step--; } } return -1; } int f[] = {11,21,31,41}; int main() { int T; scanf("%d", &T); while(T--) { cur = 0; for(int i = 0; i < 4; ++i) for(int j = 0; j < 8; ++j) { if(j == 0) st[i][j] = 0; else scanf("%d", &st[i][j]); } for(int i = 0; i < 4; ++i) { for(int j = 1; j < 8; ++j) { switch(st[i][j]) { case 11: swap(st[i][j], st[0][0]); sp[0][0] = i, sp[0][1] = j; break; case 21: swap(st[i][j], st[1][0]); sp[1][0] = i, sp[1][1] = j; break; case 31: swap(st[i][j], st[2][0]); sp[2][0] = i, sp[2][1] = j; break; case 41: swap(st[i][j], st[3][0]); sp[3][0] = i, sp[3][1] = j; break; } } } printf("%d\n", bfs()); } return 0; }
如有不当之处欢迎指出!
HDU - 1067 Gap (bfs + hash) [kuangbin带你飞]专题二的更多相关文章
- 【算法系列学习三】[kuangbin带你飞]专题二 搜索进阶 之 A-Eight 反向bfs打表和康拓展开
[kuangbin带你飞]专题二 搜索进阶 之 A-Eight 这是一道经典的八数码问题.首先,简单介绍一下八数码问题: 八数码问题也称为九宫问题.在3×3的棋盘,摆有八个棋子,每个棋子上标有1至8的 ...
- hdu.1067.Gap(bfs+hash)
Gap Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Subm ...
- HDU - 3533 bfs [kuangbin带你飞]专题二
看了好久的样例才看懂. 题意:有一个人要从(0,0)走到(n,m),图中有k个碉堡,每个碉堡可以向某个固定的方向每隔t秒放一次炮,炮弹不能穿越另一个碉堡,会被阻挡.人在移动的过程中不会被炮弹打到,也就 ...
- HDU - 3085 双向BFS + 技巧处理 [kuangbin带你飞]专题二
题意:有两只鬼,一个男孩女孩被困在迷宫中,男孩每秒可以走三步,女孩只能1步,鬼可以两步且可以通过墙.问男孩女孩是否可以在鬼抓住他们之前会合? 注意:每秒开始鬼先移动,然后两人开始移动. 思路:以男孩和 ...
- HDU - 2102 A计划 (BFS) [kuangbin带你飞]专题二
思路:接BFS判断能否在限制时间内到达公主的位置,注意如果骑士进入传送机就会被立即传送到另一层,不会能再向四周移动了,例如第一层的位置(x, y, 1)是传送机,第二层(x, y, 2)也是传送机,这 ...
- HDU - 3567 Eight II (bfs预处理 + 康托) [kuangbin带你飞]专题二
类似HDU1430,不过本题需要枚举X的九个位置,分别保存状态,因为要保证最少步数.要保证字典序最小的话,在扩展节点时,方向顺序为:down, left, right, up. 我用c++提交1500 ...
- 【算法系列学习】[kuangbin带你飞]专题二 搜索进阶 D - Escape (BFS)
Escape 参考:http://blog.csdn.net/libin56842/article/details/41909459 [题意]: 一个人从(0,0)跑到(n,m),只有k点能量,一秒消 ...
- HDU - 1043 A* + 康托 [kuangbin带你飞]专题二
这题我第一次用的bfs + ELFhash,直接TLE,又换成bfs + 康托还是TLE,5000ms都过不了!!我一直调试,还是TLE,我才发觉应该是方法的问题. 今天早上起床怒学了一波A*算法,因 ...
- HDU - 3001 Travelling 状压dp + 三进制 [kuangbin带你飞]专题二
终于刷完搜索专题了. 题意:给定n个城市,每个城市参观不能超过两次,两个城市之间有道路通过需要花费X,求通过能所有城市的最小花费. 思路:每个城市有三个状态0,1,2,可用三进制存储所有城市的访问状态 ...
随机推荐
- MySQL复制相关变量
server_id是必须设置在master和每个slave上的唯一标识ID,其取值范围 是1~4294967295之间,且同一个复制组之内不能重复 server_uuid:server_uuid会在G ...
- 【转】国际GNSS服务组织IGS
国际GNSS服务组织The International GNSS Service,简称IGS,前身为国际GPS服务组织.IGS提供的高质量数据和产品被用于地球科学研究等多个领域. IGS组织由卫星跟踪 ...
- mysql数据库在Linux下安装与配置
mysql是一种开放源代码的关系型数据库系统(RDBMS),使用最常用的数据库管理语言--结构化查询语句(SQL)进行数据库管理. 由于mysql的开源,因此任何人都可以在General Publ ...
- unicode文件处理(如果是ANSI编码就不需要了)
1.unicode文件的打开必须用rb模式. 3.wchar_t str[100] = { 0 }; 这个占200个字节. 2.宽字符对应的处理 fgetc fgetwc fputc fputwc f ...
- Global对象
1.Global对象是不需要创建实例就可以直接调用方法或属性的对象.有点类是于java中的工具类 2.关于汉字的网络传递 网络访问的过程中在传递数据的时候,传递的本质都是0101,如果你要传送字符的话 ...
- Electron应用使用electron-builder配合electron-updater实现自动更新(windows + mac)
发客户端一定要做的就是自动更新模块,否则每次版本升级都是一个头疼的事.下面是Electron应用使用electron-builder配合electron-updater实现自动更新的解决方案. 1.安 ...
- Django之wagtail安装及配置
安装指引原文地址:Http://docs.wagtail.io/en/v1.13.1 需要注意的几点: 指定端口启动服务:在项目根目录下 ,运行 python manage.py runserver ...
- CAS基础和原子类
基于CAS实现的AtomicInteger. AtomicLong. AtomicReference. AtomicBoolean也被称为乐观锁. CAS的语义是“我认为V的值应该为A,如果是,那么将 ...
- 洛谷 [P1387] 最大正方形
本题非常有趣. (n^6) 枚举四个端点,每次遍历矩阵求解. (n^4) 先处理前缀和,枚举四个端点,每次比较前缀和和正方形面积. (n^3) 枚举左上方端点,在枚举边长,前缀和优化 (n^2logn ...
- Nodejs的运行原理-函数回调篇
前言 当客户端向http server 发起TCP链接时,server端会发起一系列的callback调用,这是一个逆向调用的过程:开始于libuv,终止于js代码里的callback(promise ...