题目链接

题意

中文题意

思路

做这题的前置技能学习

  1. 康托展开

    这个东西我认为就是在排列组合问题上的Hash算法,可以压缩空间。

  2. A*搜索。

    这里我使用了像k短路一样的做法,从最终状态倒回去预处理一遍距离,但是跑了0.8s,可能是预处理花费的时间太多了。有些人用曼哈顿距离估价,跑了0.2s。

#include <bits/stdc++.h>
using namespace std;
const int INF = 0x3f3f3f3f;
const int N = 1e5 + 10;
typedef long long LL;
struct Node {
int x, y, f, g, kt, mm[4][4];
bool operator < (const Node &rhs) const {
return f > rhs.f;
}
};
int init[4][4] = {
{ 1, 2, 3 },
{ 4, 5, 6 },
{ 7, 8, 0 }
};
int mp[4][4], ans, h[500001], vis[500001], target;
int f[10], dx[] = {0, 0, 1, -1}, dy[] = {1, -1, 0, 0}; int kangtuo(int mp[4][4]) {
int sum = 0;
for(int i = 0; i < 9; i++) {
int now = 0;
for(int j = i + 1; j < 9; j++) {
if(mp[i/3][i%3] > mp[j/3][j%3]) now++;
}
sum += now * f[9-i-1];
} return sum + 1;
} void BFS(int x, int y) {
memset(h, INF, sizeof(h));
queue<Node> que;
Node now = (Node) {x, y, 0, 0};
for(int i = 0; i < 3; i++) for(int j = 0; j < 3; j++)
now.mm[i][j] = init[i][j];
h[target] = 0;
que.push(now);
while(!que.empty()) {
Node now = que.front(); que.pop();
int x = now.x, y = now.y, f = now.f;
for(int i = 0; i < 4; i++) {
now.x = x + dx[i], now.y = y + dy[i];
if(now.x < 0 || now.x > 2 || now.y < 0 || now.y > 2) continue;
swap(now.mm[x][y], now.mm[now.x][now.y]);
int nf = f + 1, nkt = kangtuo(now.mm);
now.f = nf;
if(h[nkt] == INF) {
h[nkt] = nf;
que.push(now);
}
swap(now.mm[x][y], now.mm[now.x][now.y]);
}
}
} void YCL() {
BFS(2, 2);
} int Astar(int x, int y) {
priority_queue<Node> que;
memset(vis, 0, sizeof(vis));
vis[kangtuo(mp)] = 1;
Node now = (Node) { x, y, 0, 0, kangtuo(mp)};
for(int i = 0; i < 3; i++)
for(int j = 0; j < 3; j++)
now.mm[i][j] = mp[i][j];
que.push(now);
while(!que.empty()) {
now = que.top(); que.pop();
int x = now.x, y = now.y, f = now.f, g = now.g, kt = now.kt;
if(kt == target) return f;
for(int i = 0; i < 4; i++) {
int nx = x + dx[i], ny = y + dy[i];
if(nx < 0 || nx > 2 || ny < 0 || ny > 2) continue;
swap(now.mm[x][y], now.mm[nx][ny]);
int nkt = kangtuo(now.mm);
now.x = nx, now.y = ny, now.g = g + 1, now.f = now.g + h[nkt], now.kt = nkt;
if(!vis[nkt]) {
vis[nkt] = 1;
que.push(now);
}
swap(now.mm[x][y], now.mm[nx][ny]);
}
} return -1;
} int main() {
f[0] = 1;
for(int i = 1; i < 9; i++) f[i] = f[i-1] * (i);
target = kangtuo(init);
YCL();
int t; scanf("%d", &t);
while(t--) {
int ix, iy;
for(int i = 0; i < 3; i++)
for(int j = 0; j < 3; j++) {
scanf("%d", &mp[i][j]);
if(mp[i][j] == 0) ix = i, iy = j;
}
ans = Astar(ix, iy);
if(~ans) printf("%d\n", ans);
else puts("No Solution!");
}
return 0;
}

hihoCoder 1312:搜索三·启发式搜索(A* + 康托展开)的更多相关文章

  1. hihoCoder #1312 : 搜索三·启发式搜索(A*, 康托展开)

    原题网址:http://hihocoder.com/problemset/problem/1312 时间限制:10000ms 单点时限:1000ms 内存限制:256MB   描述 在小Ho的手机上有 ...

  2. 【hihocoder 1312】搜索三·启发式搜索(启发式搜索写法)

    [题目链接]:http://hihocoder.com/problemset/problem/1312?sid=1092363 [题意] [题解] 定义一个A*函数 f = step+val 这里的v ...

  3. 【hihocoder 1312】搜索三·启发式搜索(普通广搜做法)

    [题目链接]:http://hihocoder.com/problemset/problem/1312?sid=1092352 [题意] [题解] 从末状态的123456780开始逆向搜; 看它能到达 ...

  4. hdu 1430(BFS+康托展开+映射+输出路径)

    魔板 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submiss ...

  5. 康托展开:对全排列的HASH和还原,判断搜索中的某个排列是否出现过

    题目:http://acm.hrbust.edu.cn/index.php?m=ProblemSet&a=showProblem&problem_id=2297 前置技能:(千万注意是 ...

  6. 双向广搜+hash+康托展开 codevs 1225 八数码难题

    codevs 1225 八数码难题  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond   题目描述 Description Yours和zero在研究A*启 ...

  7. HDU 1430 魔板(康托展开+BFS+预处理)

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

  8. hdu1430魔板(BFS+康托展开)

    做这题先看:http://blog.csdn.net/u010372095/article/details/9904497 Problem Description 在魔方风靡全球之后不久,Rubik先 ...

  9. poj1077(康托展开+bfs+记忆路径)

    题意:就是说,给出一个三行三列的数组,其中元素为1--8和x,例如: 1 2 3 现在,需要你把它变成:1 2 3 要的最少步数的移动方案.可以右移r,左移l,上移u,下移dx 4 6 4 5 67 ...

随机推荐

  1. WPF编游戏系列 之三 物品清单

    原文:WPF编游戏系列 之三 物品清单        本篇将介绍如何通过C#自动生成游戏界面,主要演示点击"My Shop"后如何显示所有物品清单.其中数据源来自于Access 2 ...

  2. RHEL 6和RHEL 7(CentOS 6和CentOS 7)恢复ROOT密码

    RedHat 6恢复Root密码: 1.启动RedHat 6的时候在这个界面按任意键 2.出现如下界面,按 e 3.出现如下界面,选择第二个--kernel,然后再按 e 4.出现如下界面,输入 空格 ...

  3. Hermite曲线插值

    原文 Hermite Curve Interpolation Hermite Curve Interpolation Hamburg (Germany), the 30th March 1998. W ...

  4. xadmin下修改左道航的显示不是中文字修改方法

    解决方案: 在对应的apps下有一个apps.py文件添加verbose_name=u"想要的字" 在对应的__init__.py 添加   default_app_config= ...

  5. vs2017 cordova apk 第一个项目

    原文:vs2017 cordova apk 第一个项目 vs出到了2017,终于能正了八经跨平台开发,特别是终于不报一堆错了. cordova是个好东西,终于不用揽一个项目,还要被手机端瓜分大半血汗钱 ...

  6. delphi 获取大于2G的物理内存大小

    一般情况下,我们是用GlobalMemoryStatus 来获取物理内存大小的 但该API在物理内存大小超过2G的时候,返回值均为2GB.因此,没有办法获取真实的物理内存大小,所以需要对此进行改进. ...

  7. 高手问答精选:Go 语言 —— 云计算时代的 C 语言(类似于一个FAQ)

    Go 语言被称为云计算时代的 C 语言,它在软件开发效率和运行效率之间做出了绝佳的权衡.这使得它既适应于互联网应用的极速开发,又能在高并发.高性能的开发场景中如鱼得水.正因如此,许多互联网公司,尤其是 ...

  8. 利用NPOI生成DOCX文档

    首先安装NPOI控件: Install-Package NPOI 代码: using NPOI.OpenXmlFormats.Wordprocessing; using NPOI.XWPF.UserM ...

  9. NAudio的使用说明

    官方网站:http://naudio.codeplex.com/ 源码:https://github.com/naudio/NAudio NuGet安装: Install-Package NAudio ...

  10. Win10自带应用不喜欢?一条命令全部卸载

    与Win8/Win8.1一样,Win10中继续内置了应用商店,所不同的是Windows10中已升级为通用应用商店,具有跨平台特性.可能有的朋友仍不喜欢使用Modern应用,因为传统桌面应用几乎能够满足 ...