Description

汉诺塔由三根柱子(分别用A B C表示)和n个大小互不相同的空心盘子组成。一开始n个盘子都摞在柱子A上,大的在下面,小的在上面,形成了一个塔状的锥形体。

对汉诺塔的一次合法的操作是指:从一根柱子的最上层拿一个盘子放到另一根柱子的最上层,同时要保证被移动的盘子一定放在比它更大的盘子上面(如果移 动到空柱子上就不需要满足这个要求)。我们可以用两个字母来描述一次操作:第一个字母代表起始柱子,第二个字母代表目标柱子。例如,AB就是把柱子A最上 面的那个盘子移到柱子B。汉诺塔的游戏目标是将所有的盘子从柱子A移动到柱子B或柱子C上面。有一种非常简洁而经典的策略可以帮助我们完成这个游戏。首 先,在任何操作执行之前,我们以任意的次序为六种操作(AB、AC、BA、BC、CA和CB)赋予不同的优先级,然后,我们总是选择符合以下两个条件的操 作来移动盘子,直到所有的盘子都从柱子A移动到另一根柱子:(1)这种操作是所有合法操作中优先级最高的;(2)这种操作所要移动的盘子不是上一次操作所 移动的那个盘子。可以证明,上述策略一定能完成汉诺塔游戏。现在你的任务就是假设给定了每种操作的优先级,计算按照上述策略操作汉诺塔移动所需要的步骤 数。

Input

输入有两行。第一行为一个整数n(1≤n≤30),代表盘子的个数。第二行是一串大写的ABC字符,代表六种操作的优先级,靠前的操作具有较高的优先级。每种操作都由一个空格隔开。

Output

只需输出一个数,这个数表示移动的次数。我们保证答案不会超过10的18次方。

Sample Input

3
AB BC CA BA CB AC

Sample Output

7

分析

网上好多题解都说暴力模拟几轮直接推出线性递推的系数直接做快速幂……可惜我太弱了不会证明这个dp方程“一定会形成一个线性递推式”……

我的做法是:设f(t, i)为将t柱上最上面的i个盘子转移到“另一个柱子”需要的操作次数,再维护一个To(t,i)表示此时t柱上的i个盘子最先整体到达的柱子编号。To数组的边界可以在读入时由操作的优先级得出。

转移方程也不难想到:

f[][] = f[][] = f[][] = ;//f数组边界
for(i = ;i < ;++i){//To数组边界
while((a = getchar()) < 'A' || a > 'C');b = getchar();
    if(!known[a-'A']){
        To[a-'A'] = b - 'A';
        known[a-'A'] = ;
    }
}
for(i = ;i <= N;++i){
    for(j = ;j < ;++j){
        tmpf[j] = f[j] +  + f[To[j]];
        if(To[To[j]] == j){
            tmpf[j] += f[j] + ;
            tmpT[j] = To[j];
        }
        else tmpT[j] = To[To[j]];
    }
    for(j = ;j < ;++j)f[j] = tmpf[j], To[j] = tmpT[j];
}

考虑到第二位为 i 时的答案只与第i-1位有关,我们可以把数组缩成一维(这样就只剩三位了好吗……)

 1 /**************************************************************
 2     Problem: 1019
 3     User: AsmDef
 4     Language: C++
 5     Result: Accepted
 6     Time:0 ms
 7     Memory:804 kb
 8 ****************************************************************/
 9  
 //#include <cctype>
 #include <cstdio>
 //#include <iostream>
 //#include <cmath>
 //#include <cstdlib>
 //#include <algorithm>
 //#include <ctime>
 //#include <assert.h>
 using namespace std;
 /*template<typename T>inline void getd(T &x){
     char c = getchar(); bool minus = 0;
     while(!isdigit(c) && c != '-')c = getchar();
     if(c == '-')minus = 1, c = getchar();
     x = c - '0';
     while(isdigit(c = getchar()))x = x * 10 + c - '0';
     if(minus)x = -x;
 }*/
 /*========================================================*/
 typedef long long LL;
 LL f[];
 int To[], N;
  
 int main(){
     #if defined DEBUG
     freopen("test", "r", stdin);
     //freopen("out.txt", "w", stdout);
     #else
     #ifndef ONLINE_JUDGE
     freopen("boxes.in", "r", stdin);
     freopen("boxes.out", "w", stdout);
     #endif
     #endif
     scanf("%d", &N);
     if(N == ){printf("1\n");return ;}
     char a, b;
     int i, j, tmpT[];
     bool known[] = {};
     f[] = f[] = f[] = ;
     for(i = ;i < ;++i){
         while((a = getchar()) < 'A' || a > 'C');b = getchar();
         if(!known[a-'A']){
             To[a-'A'] = b - 'A';
             known[a-'A'] = ;
         }
     }
     LL tmpf[];
     for(i = ;i <= N;++i){
         for(j = ;j < ;++j){
             tmpf[j] = f[j] +  + f[To[j]];
             if(To[To[j]] == j){
                 tmpf[j] += f[j] + ;
                 tmpT[j] = To[j];
             }
             else tmpT[j] = To[To[j]];
         }
         for(j = ;j < ;++j)f[j] = tmpf[j], To[j] = tmpT[j];
     }
     printf("%lld\n", f[]);
      
     #ifdef DEBUG
     //cout << endl << (double)clock() / CLOCKS_PER_SEC << endl;
     #endif
     return ;
 }
 

动态规划

[bzoj1019][SHOI2008]汉诺塔 (动态规划)的更多相关文章

  1. bzoj1019: [SHOI2008]汉诺塔(动态规划)

    1019: [SHOI2008]汉诺塔 题目:传送门 简要题意: 和经典的汉诺塔问题区别不大,但是题目规定了一个移动时的优先级: 如果当前要从A柱子移动,但是A到C的优先级比A到B的优先级大的话,那就 ...

  2. bzoj1019 [SHOI2008]汉诺塔

    1019: [SHOI2008]汉诺塔 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 1030  Solved: 638[Submit][Status] ...

  3. bzoj千题计划109:bzoj1019: [SHOI2008]汉诺塔

    http://www.lydsy.com/JudgeOnline/problem.php?id=1019 题目中问步骤数,没说最少 可以大胆猜测移动方案唯一 (真的是唯一但不会证) 设f[i][j] ...

  4. 【BZOJ1019】[SHOI2008]汉诺塔(数论,搜索)

    [BZOJ1019][SHOI2008]汉诺塔(数论,搜索) 题面 BZOJ 洛谷 题解 首先汉诺塔问题的递推式我们大力猜想一下一定会是形如\(f_i=kf_{i-1}+b\)的形式. 这个鬼玩意不好 ...

  5. bzoj1019 / P4285 [SHOI2008]汉诺塔

    P4285 [SHOI2008]汉诺塔 递推 题目给出了优先级,那么走法是唯一的. 我们用$0,1,2$代表$A,B,C$三个柱子 设$g[i][x]$为第$x$根柱子上的$i$个盘子,经过演变后最终 ...

  6. 【bzoj1019】[SHOI2008]汉诺塔

    1019: [SHOI2008]汉诺塔 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 1427  Solved: 872[Submit][Status] ...

  7. BZOJ1019 汉诺塔/洛谷P4285 [SHOI2008]汉诺塔

    汉诺塔(BZOJ) P4285 [SHOI2008]汉诺塔 居然是省选题,还是DP!(我的DP菜得要死,碰见就丢分) 冥思苦想了1h+ \(\to\) ?! 就是普通的hanoi NOI or HNO ...

  8. 1019: [SHOI2008]汉诺塔

    1019: [SHOI2008]汉诺塔 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 1495  Solved: 916[Submit][Status] ...

  9. 【bzoj1019】汉诺塔

    [bzoj1019]汉诺塔 题意 传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=1019 分析 思路1:待定系数+解方程 设\(f[n]\)为 ...

随机推荐

  1. Coursera在线学习---第六节.构建机器学习系统

    备: High bias(高偏差) 模型会欠拟合    High variance(高方差) 模型会过拟合 正则化参数λ过大造成高偏差,λ过小造成高方差 一.利用训练好的模型做数据预测时,如果效果不好 ...

  2. 【Python学习】request库

    Requests库(https://www.python-requests.org/)是一个擅长处理那些复杂的HTTP请求.cookie.header(响应头和请求头)等内容的Python第三方库. ...

  3. (转)USB体系结构

    转载地址:http://blog.ednchina.com/zenhuateng/203584/Message.aspx USB总线接口层:物理连接.电气信号环境.信息包传输机制:主机一方由USB主控 ...

  4. php判断是手机还是pc访问从而走不同url

    <?php header("Content-type:text/html;charset=utf-8"); function is_mobile(){ $user_agent ...

  5. plsql实例精讲部分笔记

    转换sql: create or replace view v_sale(year,month1,month2,month3,month4,month5,month6,month7,month8,mo ...

  6. python 面试

    知识总结 面试(一)

  7. fail2ban安全设置

    1.先安装fail2ban服务包(这里我采用的是fail2ban-0.8.14.tar.gz) 2.解压安装包 cd /data/software tar xzf fail2ban-0.8.14.ta ...

  8. Jmeter配置元件执行顺序

    一.Jmeter组件 Sampler(采样器):真正干活的,比如http 定时器:默认为毫秒. 前置处理器和后置处理器:必须在有采样器才有效,在采样器的前后执行.作关联,拿响应中的数据使用,用得比较多 ...

  9. Pylint在项目中的使用

    需求背景: Pylint 是一个 Python 代码分析工具,它分析 Python 代码中的错误,查找不符合代码风格标准和有潜在问题的代码. Pylint 是一个 Python 工具,除了平常代码分析 ...

  10. Hadoop2.5.2 安装部署

    0x00 平台环境 OS: CentOS-6.5-x86_64 JDK: jdk-8u111-linux-x64 Hadoop: hadoop-2.5.2 0x01 操作系统基本设置 1.1 网络配置 ...