https://www.luogu.org/problem/show?pid=T2485

题目背景

汉诺塔升级了

题目描述

现在我们有N个圆盘和N个柱子,每个圆盘大小都不一样,大的圆盘不能放在小的圆盘上面,N个柱子从左向右排成一排。每次你可以将一个柱子的最上面的圆盘移动到右边或者左边的柱子上(如果移动是合法的话)。现在告诉你初始时的状态,你希望用最少的步数将第i大的盘子移动到第i根柱子上,问最小步数。

输入输出格式

输入格式:

第一行一个整数T,代表询问的组数。

接下来T组数据,每组数据第一行一个整数N。

接下来一行N个正整数,代表每个柱子上圆盘的大小。

输出格式:

输出共T行,代表每次的答案。如果方案不存在,输出-1;

输入输出样例

输入样例#1:

4
3
2 1 3
2
7 8
2
10000 1000
3
97 96 95
输出样例#1:

4
0
-1
20

说明

对于70%的数据,N的值都是相等的。

对于100%的数据,1≤T≤6 x 10^3 ; 1≤N≤7。

其实这个题其实是打表啊…..用BFS打表,然后对应询问直接输出,时空复杂度大丈夫。

对于读入进来的数据,离散化一下,用z数组记录这个盘子是第几大。

用bit数组记录位,这个是当你移动盘子的时候有用的….(具体我也说不太清楚,看代码理解理解吧。) 
我们开一个top和place数组,分别记录第i个柱子的最上面的盘子,和当前盘子在哪根柱子上, 
预处da理biao的时候,首先从初始状态开始搜(1,12,123,1234……1234567) 
每搜到一个新的可能状态,就把这个状态记录下来。(试想一下,从某个给定状态到初始状态的步数==从初始状态的步数到给定状态的步数) 
细节处理的话,直接看代码吧。

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
const int maxn=;
int n,top[],place[],bit[];//最上面|每个盘子位置 |进制保存
int res[maxn];//答案
int q[maxn];//队列 int front,tail;//队列头尾指针
int w[],z[];//z为离散化后的大小
bool use[maxn];
bool cmp(int a,int b)
{
return w[a]<w[b];
}//按大小排序
void analyze(int s)
{
int x=;
int ss=s;
for (int a=;a<=;a++)
top[a]=;//最上面的
while (ss)
{
x++;
place[x]=ss%;//初始盘子位置(处理是反着的)
ss/=;
}
reverse(place+,place+x+);//逆转 (一个奇怪的函数) for (int a=x;a>=;a--)
top[place[a]]=a;
//初始最上面位置 即1 2 3 4 5 6 7 for (int a=;a<=x;a++)
{
if (a==top[place[a]])//a在某个柱子的最上面//移动的时候,是直接往左移动一个柱子或直接往右移一个柱子
{
int p=place[a];//那个柱
if (p!= && (a<top[p-] || !top[p-]))//小或者空
{
int news=s-bit[x-a];//新状态
if (!use[news])
{
q[++tail]=news;
use[news]=true;
res[news]=res[s]+;//因为是宽搜,第一次搜到的就是最小的。
}
}//左移
if (p!=x && (a<top[p+] || !top[p+]))
{
int news=s+bit[x-a];
if (!use[news])
{
q[++tail]=news;//orz 钟小鸟
use[news]=true;
res[news]=res[s]+;
}
}//右移
}
} } int main()
{
// freopen("huakai.in","r",stdin);
// freopen("huakai.out","w",stdout);
front=,tail=;
int status=;
bit[]=;
for(int a=;a<=;a++)
{
bit[a]=bit[a-]*;
status=status*+a;//1 12 123 1234
q[++tail]=status;//1 12 134 1234
use[status]=true;//1 12 123 1234
}//bit 10 100 1000 10000...
for(;front<=tail;)
{
int s=q[front++];//
analyze(s);//
}
printf("%d",tail);
int t;
scanf("%d",&t);
for(;t--;)
{
scanf("%d",&n); for (int a=;a<=n;a++)
scanf("%d",&w[a]),z[a]=a;//按编号sort
sort(z+,z+n+,cmp); int s=;
for (int a=;a<=n;a++)
s=s*+z[a];//盘子大小序列 if (!use[s]) printf("-1\n");//移动不到
else printf("%d\n",res[s]);
}
return ;
}

T2485 汉诺塔升级版(普及)(递归)的更多相关文章

  1. 汉诺塔算法的递归与非递归的C以及C++源代码

    汉诺塔(又称河内塔)问题其实是印度的一个古老的传说. 开天辟地的神勃拉玛(和中国的盘古差不多的神吧)在一个庙里留下了三根金刚石的棒,第一根上面套着64个圆的金片,最大的一个在底下,其余一个比一 个小, ...

  2. python汉诺塔问题的递归理解

    一.问题背景 汉诺塔问题是源于印度一个古老传说. 源于印度一个古老传说的益智玩具.大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘.大梵天命令婆罗门把圆盘从下 ...

  3. C#中汉诺塔问题的递归解法

    百度测试部2015年10月份的面试题之——汉诺塔. 汉诺塔就是将一摞盘子从一个塔转移到另一个塔的游戏,中间有一个用来过度盘子的辅助塔. 百度百科在此. 游戏试玩在此. 用递归的思想解决汉诺塔问题就是分 ...

  4. [Python3 练习] 006 汉诺塔2 非递归解法

    题目:汉诺塔 II 接上一篇 [Python3 练习] 005 汉诺塔1 递归解法 这次不使用递归 不限定层数 (1) 解决方式 利用"二进制" (2) 具体说明 统一起见 我把左 ...

  5. PTA 汉诺塔的非递归实现(C 语言)

    借助堆栈以非递归(循环)方式求解汉诺塔的问题(n, a, b, c), 即将N个盘子从起始柱(标记为“a”)通过借助柱(标记为“b”)移动到目标柱(标记为“c”), 并保证每个移动符合汉诺塔问题的要求 ...

  6. 汉诺塔算法c++源代码(递归与非递归)[转]

     算法介绍: 其实算法非常简单,当盘子的个数为n时,移动的次数应等于2^n - 1(有兴趣的可以自己证明试试看).后来一位美国学者发现一种出人意料的简单方法,只要轮流进行两步操作就可以了.首先把三根柱 ...

  7. python数据结构_递归_汉诺塔问题

    已经不是第一次写这个汉诺塔问题, 其实递归还真是不太好理解, 因为递归这种是想其实有点反人类, 为什么? 因为不太清楚, 写个循环一目了然, 用递归其实要把核心逻辑理清楚, 要不根本没法进行下去 所有 ...

  8. [Python3 练习] 005 汉诺塔1 递归解法

    题目:汉诺塔 I (1) 描述 传说,在世界中心贝拿勒斯(在印度北部)的圣庙外有左中右三根足够长的柱子(塔) 左边柱子上套着 64 片金片,金片按"上小下大"排,其余两根是空柱子 ...

  9. 汉诺塔问题C++实现

    大家好,我是小鸭酱,博客地址为:http://www.cnblogs.com/xiaoyajiang 以下进行汉诺塔问题的递归实现 #include <iostream.h> int gb ...

随机推荐

  1. 2018-8-10-使用-ahk-让普通键盘变为Dvorak键盘

    title author date CreateTime categories 使用 ahk 让普通键盘变为Dvorak键盘 lindexi 2018-08-10 19:16:51 +0800 201 ...

  2. passwd的使用例子

    passwd 作为普通用户和超级权限用户都可以运行,但作为普通用户只能更改自己的用户密码,但前提是没有被root用户锁定:如果root用户运行passwd ,可以设置或修改任何用户的密码: passw ...

  3. 洛谷P3834【模板】可持久化线段树 1(主席树)

    题目背景 这是个非常经典的主席树入门题--静态区间第K小 数据已经过加强,请使用主席树.同时请注意常数优化 题目描述 如题,给定N个正整数构成的序列,将对于指定的闭区间查询其区间内的第K小值. 输入输 ...

  4. [转]Git 常用命令详解

    史上最浅显易懂的Git教程 http://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000/ ht ...

  5. leetcode-119-杨辉三角②

    题目描述: 第一次提交: class Solution: def getRow(self, rowIndex: int) -> List[int]: k = rowIndex pre = [1] ...

  6. 来杭州云栖大会,全面了解企业如何实现云上IT治理

    企业上云的现状与趋势 云计算,如今已经成为了像水和电一般关系到国计民生的国家基础设施.云计算为企业带了前所未有的资源交付效率和运维效率的提升,同时也用全新的技术帮助企业在新的价值网络中创造新的商业赛道 ...

  7. Ubuntu GitHub操作——分支、合并与标签

    分支 分支是用来将特性开发绝缘开来的.在你创建仓库的时候,master 是"默认的"分支.在其他分支上进行开发,完成后再将它们合并到主分支上. 创建一个叫做"featur ...

  8. html 引入公共的头部和底部

  9. 19-10-30-C

    交文件吼啊. ZJ一下: T1是真·高中数学. T2不是很清楚,只得了30. T3打了一个欧拉序. 做的海星的地方: Vim太好用辣,直接按平常打叫上去它就是 freopen T1仔仔细细的研究了高考 ...

  10. PAT甲级——A1004 Counting Leaves

    A family hierarchy is usually presented by a pedigree tree. Your job is to count those family member ...