题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6171

题意: 给你一个高度为6的塔形数组,你每次只能将0与他上下相邻的某个数交换,问最少交换多少次可以变为初始状态,若需要的步数大于20,直接输出too difficult,初始状态为:


1 1 
2 2 2 
3 3 3 3 
4 4 4 4 4 
5 5 5 5 5 5

解法:两种方法,一种是双向BFS+Hash,另外是A*估价+Hash。

双向搜索参考:http://blog.csdn.net/cillyb/article/details/77587228

//双BFS
#include <bits/stdc++.h>
using namespace std;
const int maxn = 100010;
const int dir[4][2] = {{-1,-1},{-1,0},{1,0},{1,1}};
typedef unsigned long long uLL;
struct node{
int val[6][6];
int r, c, k, step;
};
map <uLL, int> book[2];
uLL Hash(node x){
uLL ret = 0;
for(int i=0; i<6; i++)
for(int j=0; j<=i; j++)
ret = ret*6+x.val[i][j];
return ret;
}
int BFS(node s, node e){
queue <node> q;
book[0].clear();
book[1].clear();
s.k = 0;
e.k = 1;
s.step = e.step = 0;
book[s.k][Hash(s)] = 0;
book[e.k][Hash(e)] = 0;
q.push(s);
q.push(e);
while(q.size()){
node u = q.front(); q.pop();
uLL tmp = Hash(u);
if(book[!u.k].count(tmp)){
if(book[!u.k][tmp]+u.step<=20){
return book[!u.k][tmp]+u.step;
}
else continue;
}
if(u.step >= 10) continue;
for(int i=0; i<4; i++){
node t = u;
t.r += dir[i][0];
t.c += dir[i][1];
if(t.r >= 6 || t.r < 0 || t.c > t.r || t.c < 0) continue;
swap(t.val[t.r][t.c], t.val[u.r][u.c]);
tmp = Hash(t);
if(book[t.k].count(tmp)) continue;
t.step++;
book[t.k][tmp] = t.step;
q.push(t);
}
}
return -1;
}
int main()
{
int T;
scanf("%d", &T);
while(T--)
{
node s,e;
e.r = e.c = 0;
for(int i=0; i<6; i++){
for(int j=0; j<=i; j++){
scanf("%d", &s.val[i][j]);
if(!s.val[i][j])
s.r = i, s.c = j;
e.val[i][j] = i;
}
}
int ans = BFS(s, e);
if(ans == -1) puts("too difficult");
else printf("%d\n", ans);
}
return 0;
}

放一个队友的A*搜索:

#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long uLL;
int lowPos[6][6];
pair<int, int> ppPos[22];
const int dir[4][2] = {{-1, -1}, {-1, 0}, {1, 0}, {1, 1}};
void init()
{
int jsq = 0;
int now = 0;
for(int i = 0; i < 6; i++)
for(int j = 0; j <= i; j++)
{
lowPos[i][j] = now;
ppPos[jsq] = make_pair(i, j);
jsq ++;
now += 3;
}
}
void setStatus(uLL &status, int x, int y, uLL val)
{
int pos = lowPos[x][y];
status &= ~(1ULL << pos);
status &= ~(1ULL << pos + 1);
status &= ~(1ULL << pos + 2);
status |= val << pos;
return;
}
void swapStatus(uLL &status, int a, int b, int x, int y)
{
int pos1 = lowPos[a][b];
int pos2 = lowPos[x][y];
uLL val1 = status >> pos1 & 7;
uLL val2 = status >> pos2 & 7;
status &= ~(1ULL << pos1);
status &= ~(1ULL << pos1 + 1);
status &= ~(1ULL << pos1 + 2);
status &= ~(1ULL << pos2);
status &= ~(1ULL << pos2 + 1);
status &= ~(1ULL << pos2 + 2);
status |= val1 << pos2;
status |= val2 << pos1;
}
pair<int, int> getZeroPos(uLL status)
{
for(int i = 0; i < 21; i++)
{
uLL val = status & 7;
if(val == 0)
return ppPos[i];
status >>= 3;
}
}
int Compare(uLL endStatus, uLL nowStatus)
{
int ret = 0;
for(int i = 0; i < 21; i++)
{
uLL val1 = endStatus & 7;
uLL val2 = nowStatus & 7;
if(val1 != val2)
ret++;
endStatus >>= 3;
nowStatus >>= 3;
}
return ret;
}
int main()
{
init();
int T;
scanf("%d", &T);
while(T--)
{
uLL startStatus = 0;
for(int i = 0; i < 6; i++)
for(int j = 0; j <= i; j++)
{
int x;
scanf("%d", &x);
setStatus(startStatus, i, j, x);
}
uLL endStatus = 0;
for(int i = 0; i < 6; i++)
for(int j = 0; j <= i; j++)
setStatus(endStatus, i, j, i);
queue<uLL> q;
unordered_map<uLL, int> mp;
mp[startStatus] = 0;
q.emplace(startStatus);
while(!q.empty())
{
uLL u = q.front();
q.pop();
if(u == endStatus)
break;
int d = mp[u];
if(d == 20)
break;
pair<int, int> zpos = getZeroPos(u);
int i = zpos.first;
int j = zpos.second;
for(int k = 0; k < 4; k++)
{
int x = i + dir[k][0];
int y = j + dir[k][1];
if(x < 0 || x > 5 || y < 0 || y > x)
continue;
uLL newStatus = u;
swapStatus(newStatus, i, j, x, y);
if(mp.count(newStatus))
continue;
mp[newStatus] = d + 1;
int famly = Compare(endStatus, newStatus);
if(famly + d + 2 > 20)
continue;
q.emplace(newStatus);
}
}
if(mp.count(endStatus) == 0)
puts("too difficult");
else
printf("%d\n", mp[endStatus]);
}
return 0;
}

2017多校第10场 HDU 6171 Admiral 双向BFS或者A*搜索的更多相关文章

  1. 2017多校第10场 HDU 6178 Monkeys 贪心,或者DP

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6178 题意:给出一棵有n个节点的树,现在需要你把k只猴子放在节点上,每个节点最多放一只猴子,且要求每只 ...

  2. 2017多校第10场 HDU 6181 Two Paths 次短路

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6181 题意:给一个图,求出次短路. 解法:我之前的模板不能解决这种图,就是最短路和次短路相等的情况,证 ...

  3. 2017多校第10场 HDU 6180 Schedule 贪心,multiset

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6180 题意:给了一些任务的开始时间和终止时间,现在让我们安排k台及机器,让这些任务在k太机器上最小,并 ...

  4. 2017多校第10场 HDU 6172 Array Challenge 猜公式,矩阵幂

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6172 题意:如题. 解法: #include <bits/stdc++.h> using ...

  5. 2017多校第9场 HDU 6161 Big binary tree 思维,类似字典树

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6161 题意: 题目是给一棵完全二叉树,从上到下从左到右给每个节点标号,每个点有权值,初始权值为其标号, ...

  6. 2017多校第9场 HDU 6169 Senior PanⅡ 数论,DP,爆搜

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6169 题意:给了区间L,R,求[L,R]区间所有满足其最小质数因子为k的数的和. 解法: 我看了这篇b ...

  7. 2017多校第8场 HDU 6138 Fleet of the Eternal Throne AC自动机或者KMP

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6138 题意:给n个串,每次询问x号串和y号串的最长公共子串的长度,这个子串必须是n个串中某个串的前缀 ...

  8. 2017多校第8场 HDU 6134 Battlestation Operational 莫比乌斯反演

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6134 题意: 解法: 那么g(n)怎么求,我们尝试打表发现g(n)是有规律的,g(n)=g(n-1)+ ...

  9. 2017多校第7场 HDU 6129 Just do it 找规律

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6129 题意:求a序列后m次xor前缀和 解法: 手动对1位置对每个位置的贡献打表发现 第一次 贡献为 ...

随机推荐

  1. Ubuntu14.04桌面系统允许root登录

    首先安装完系统后,在登录界面我们可以看到不允许root账户登录.以普通账户登录系统,打开终端.执行如下命令来设置root密码: sudo passwd root 然后执行命令修改如下配置文件: vi ...

  2. ReactiveSwift源码解析(九) SignalProducerProtocol延展中的Start、Lift系列方法的代码实现

    上篇博客我们聊完SignalProducer结构体的基本实现后,我们接下来就聊一下SignalProducerProtocol延展中的start和lift系列方法.SignalProducer结构体的 ...

  3. python爬虫从入门到放弃(一)之初识爬虫

    整理这个文档的初衷是自己开始学习的时候没有找到好的教程和文本资料,自己整理一份这样的资料希望能对小伙伴有帮助 什么是爬虫? 网络爬虫(又被称为网页蜘蛛,网络机器人,在FOAF社区中间,更经常的称为网页 ...

  4. Java栈与堆 (转)

    1. 栈(stack)与堆(heap)都是Java用来在Ram中存放数据的地方.与C++不同,Java自动管理栈和堆,程序员不能直接地设置栈或堆. 2. 栈的优势是,存取速度比堆要快,仅次于直接位于C ...

  5. ubuntu中vim下按上下左右键时输入A、B、C、D的问题

    ubuntu下使用vi 进行编辑文件时,按上下左右键时,会输入A.B.C.D,这个用起来很不方便.网上查得此问题的原因是: ubuntu系统自带的 vi 不完整导致,解决方法:安装完整的vi $ su ...

  6. 强连通分量tarjan缩点——POJ2186 Popular Cows

    这里的Tarjan是基于DFS,用于求有向图的强联通分量. 运用了一个点dfn时间戳和low的关系巧妙地判断出一个强联通分量,从而实现一次DFS即可求出所有的强联通分量. §有向图中, u可达v不一定 ...

  7. Hadoop - 国内各站点最高温度、气压和风速统计

    版权说明:  本文章版权归本人及博客园共同所有,转载请标明原文出处(http://www.cnblogs.com/mikevictor07/),以下内容为个人理解,仅供参考. 一.简介 该实例统计国内 ...

  8. UE4 C++BeginPlay And BlueprintBeginPlay

    今天遇到了一个诡异的问题,经过几个小时的煎熬终于找到了原因.mmmp 如果有一个类AActorChild,这个AActorChild继承自AActor,再有一个蓝图类BPAActorChild. 蓝图 ...

  9. 简单探讨python中的语句和语法

    python程序结构 python"一切皆对象",这是接触python听到最多的总结了.在python中最基层的单位应该就是对象了,对象需要靠表达式建立处理,而表达式往往存在于语句 ...

  10. C语言库函数探究

    1.strlen()求字符串长度 //模拟实现strlen函数 #include<stdio.h> #include<stdlib.h> #include<string. ...