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. mybatis框架中XxxxMaper.xml的文件

    我们知道在mybatis框架中,config.xml中会关联到许多的XxxxMapper的xml文件,这些文件又对应着一个个的接口,来观察下这些xml文件 从以下这个文件为例子: <?xml v ...

  2. 03_mybatis配置文件详解

    1. SqlMapConfig.xml mybatis全局配置文件SqlMapConfig.xml,配置内容如下: *properties(属性) setting(全局配置参数) typeAliase ...

  3. opensuse 通过composer安装drush工具

    由于笔者的opensuse已安装好composer,所以按照官方网站的文章 Installing/Upgrading Drush on Ubuntu,使用composer形式安装drush工具. co ...

  4. JUC 一 CopyOnWriteArrayList 和 CopyOnWriteArraySet

    java.util.concurrent; 简介 CopyOnWriteArrayList是一个线程安全的ArrayList,通过内部的volatile数组和显式锁ReentrantLock来实现线程 ...

  5. HAVING方法也是连贯操作之一

    HAVING方法也是连贯操作之一,用于配合group方法完成从分组的结果中筛选(通常是聚合条件)数据. having方法只有一个参数,并且只能使用字符串,例如: $this->field('us ...

  6. duilib教程之duilib入门简明教程1.前言

    关于duilib的介绍就不多讲了,一来不熟,二来小伙伴们想必已经对比了多个界面库,也无需赘述.下面进入正题:    不看广告看疗效! 已有众多知名公司采用duilib做为界面库,如华为网盘.PPS(P ...

  7. neo4j 实战、实例、示例 创建电影关系图 -1

    1. 创建关系 因为代码占篇幅太大,创建整个"电源关系图"的代码在文章最下方. 2. 简单分析创建语句 2.1 创建电影节点 CREATE (TheMatrix:Movie {ti ...

  8. Unity IoC Base On MVC

    Unity框架,是一个经典的IoC模式实现方式,其通过config文件配置section,将接口与实现解藕,config中的section配置的container以全名称对应,使得应用程序无需像Nin ...

  9. C#跨域

    //在ConfigureServices中配置 #region 跨域 var urls = "*";//Configuration["AppConfig:Cores&qu ...

  10. mysql表时间戳字段设置

    创建时间 修改时间