链接:http://poj.org/problem?id=1958

大意:汉诺塔升级版,四根柱子,n个盘子,求最少移动次数;

两种方法 递推or递归(当然还有思路3——打表)

思路1:递推(或者DP?)

把四塔转换为三塔进行思考

假设当前要移动n个盘子,那么就不如分为以下几步

先将上面的i个盘子移到第2或3个塔上;(四塔移动)

再把剩下的(n-i)个盘子移到最后一个塔上(三塔移动);

最后把在那i个盘子移到最后一个塔上(注意:是四塔移动,i个盘子一定比后来的(n-i)个盘子小)

得到方程:f[n]=min{2*f[i]+d[n-i]};

注:f[i]为四塔移动的最小步数,d[i]为三塔移动的最小步数(此处不再多说,都知道是2^i-1了,由于题面要求12,打表就行,由于两次四塔移动,一次三塔移动,所以F[N]*2)

核心代码:

for(int i=1;i<=n;i++)
{
f[n]=min(f[n],2*f[i]+d[n-i]);
}

AC代码

#include<cstdio>
#include<iostream>
#include<cctype>
#include<algorithm>
using namespace std; inline int read()
{
int ans=0,f=1;
char chr=getchar();
while(!isdigit(chr)) {if(chr='-') f=-1; chr=getchar();}
while(isdigit(chr)) {ans=ans*10+chr-'0'; chr=getchar();}
return ans*f;
}//没有读入,依然要坚持加上/笑哭
int f[20]={0};
int d[]={0,1,3,7,15,31,63,127,255,511,1023,2047,4095};
int main()
{
int n=1;
f[1]=1;
f[2]=3;
for(n;n<=12;n++)
{
if(f[n])
{
printf("%d\n",f[n]);
continue;
}
f[n]=0x3f3f3f3f;
for(int i=1;i<=n;i++)
{
f[n]=min(f[n],2*f[i]+d[n-i]);
}
printf("%d\n",f[n]);
}
return 0;
}

思路2:递归

总体思路一模一样,实现有区别而已;

#include<cstdio>
#include<iostream>
#include<cctype>
#include<cstring>
#include<algorithm>
using namespace std; inline int read()
{
int ans=0,f=1;
char chr=getchar();
while(!isdigit(chr)) {if(chr='-') f=-1; chr=getchar();}
while(isdigit(chr)) {ans=ans*10+chr-'0'; chr=getchar();}
return ans*f;
}
int f[20];
int d[]={0,1,3,7,15,31,63,127,255,511,1023,2047,4095};
int hanoi(int x)
{
if(f[x])
return f[x];//记忆化(这个算吗/晕)
if(x==0)
return 0;
if(x==1)
return f[1]=1;
f[x]=0x3f3f3f3f;
for(int i=1;i<x;i++)
f[x]=min(f[x],2*hanoi(i)+d[x-i]);//方程;
return f[x];
} int main()
{
memset(f,0,sizeof(f));
hanoi(12);
for(int i=1;i<=12;i++)
printf("%d\n",f[i]);
return 0;
}

思路3:打表(hmmmm)

可以说是最快的思路了

#include<cstdio>
#include<iostream>
#include<cctype>
#include<algorithm>
using namespace std; inline int read()
{
int ans=0,f=1;
char chr=getchar();
while(!isdigit(chr)) {if(chr='-') f=-1; chr=getchar();}
while(isdigit(chr)) {ans=ans*10+chr-'0'; chr=getchar();}
return ans*f;
}
int f[]={0,1,3,5,9,13,17,25,33,41,49,65,81};
int main()
{
for(int i=1;i<=12;i++)
printf("%d\n",f[i]);
return 0;
}

poj1958-汉诺四塔问题(三种方法)的更多相关文章

  1. 像画笔一样慢慢画出Path的三种方法(补充第四种)

    今天大家在群里大家非常热闹的讨论像画笔一样慢慢画出Path的这种效果该如何实现. 北京-LGL 博客号@ligl007发起了这个话题.然后各路高手踊跃发表意见.最后雷叔 上海-雷蒙 博客号@雷蒙之星 ...

  2. JavaScript进阶(四)js字符串转换成数字的三种方法

    js字符串转换成数字的三种方法 在js读取文本框或者其它表单数据的时候获得的值是字符串类型的,例如两个文本框a和b,如果获得a的value值为11,b的value值为9 ,那么a.value要小于b. ...

  3. Linux启动新进程的三种方法

    程序中,我们有时需要启动一个新的进程,来完成其他的工作.下面介绍了三种实现方法,以及这三种方法之间的区别. 1.system函数-调用shell进程,开启新进程system函数,是通过启动shell进 ...

  4. java 获取随机数的三种方法

    方法1(数据类型)(最小值+Math.random()*(最大值-最小值+1))例:(int)(1+Math.random()*(10-1+1))从1到10的int型随数 方法2获得随机数for (i ...

  5. 在 C# 中,(int) ,Int32.Parse() 和 Convert.toInt32() 三种方法的区别

    在 C# 中,(int),Int32.Parse() 和 Convert.toInt32() 三种方法有何区别? int 关键字表示一种整型,是32位的,它的 .NET Framework 类型为 S ...

  6. MySQL提供了以下三种方法用于获取数据库对象的元数据

    MySQL提供了以下三种方法用于获取数据库对象的元数据: 1)show语句 2)从INFORMATION_SCHEMA数据库里查询相关表 3)命令行程序,如mysqlshow, mysqldump 用 ...

  7. Java/JSP获得客户端网卡MAC地址的三种方法解析

    java/jsp获得客户端(IE)网卡MAC地址的方法大概有三种. 1.通过命令方式,在客户端执行Ipconfig 等等.(java/jsp) 2.通过ActiveX的方法.(jsp) 3.通过向13 ...

  8. Javascript 创建对象的三种方法及比较【转载+整理】

    https://developer.mozilla.org/zh-CN/docs/JavaScript/Guide/Inheritance_and_the_prototype_chain 本文内容 引 ...

  9. Ubuntu Linux系统三种方法添加本地软件库

    闲着没事教教大家以Ubuntu Linux系统三种方法添加本地软件库,ubuntu Linux使用本地软件包作为安装源——转2007-04-26 19:47新手重新系统的概率很高,每次重装系统后都要经 ...

  10. Oracle数据库备份与恢复的三种方法

    转自blueskys567原文Oracle数据库备份与恢复的三种方法, 2006-10. 有删改 Oracle数据库有三种标准的备份方法,它们分别是导出/导入(EXP/IMP).热备份和冷备份. 导出 ...

随机推荐

  1. .net core 使用 textSharp生成pdf

    引入Nuget包 using iTextSharp.text; using iTextSharp.text.pdf; using System; using System.IO; namespace ...

  2. BZOJ 3489: A simple rmq problem KDtree

    Code: #include<bits/stdc++.h> #define maxn 200000 #define inf 100000000 #define mid ((l+r)> ...

  3. Eclipse安装和使用TFS

    第一步下载Tfs插件 去微软官网下载https://www.microsoft.com/en-us/download/details.aspx?id=4240 点击 选择下载 随便放置到一个本地或者服 ...

  4. 【LeetCode】1、Two Sum

    题目等级:Easy 题目描述:   Given an array of integers, return indices of the two numbers such that they add u ...

  5. 洛谷——P1966 火柴排队&&P1774 最接近神的人_NOI导刊2010提高(02)

    P1966 火柴排队 这题贪心显然,即将两序列中第k大的数的位置保持一致,证明略: 树状数组求逆序对啦 浅谈树状数组求逆序对及离散化的几种方式及应用 方法:从前向后每次将数插入到bit(树状数组)中, ...

  6. [LUOGU]4932 浏览器

    \(\_\_stdcall\)大佬出的题\(Orz\) 我们惊奇地发现,加入\(\_\_popcount(x)\)和\(\_\_popcount(y)\)的奇偶数性相同,那么\(\_\_popcoun ...

  7. os、sys模块

    os模块 os模块是与操作系统交互的一个接口 os.makedirs("dirname1/dirname2") # 可生成多层递归目录 os.removedirs("di ...

  8. Python中的可迭代对象/迭代器/For循环工作机制/生成器

    本文分成6个部分: 1.iterable iterator区别 2.iterable的工作机制 3.iterator的工作机制 4.for循环的工作机制 5.generator的原理 6.总结 1.i ...

  9. python爬虫数据解析的四种不同选择器Xpath,Beautiful Soup,pyquery,re

    这里主要是做一个关于数据爬取以后的数据解析功能的整合,方便查阅,以防混淆 主要讲到的技术有Xpath,BeautifulSoup,PyQuery,re(正则) 首先举出两个作示例的代码,方便后面举例 ...

  10. [NOIP2016day2T1] 組合數問題(problem)

    题目描述 组合数C(n,m)表示的是从n个物品中选出m个物品的方案数.举个例子,从(1,2,3) 三个物品中选择两个物品可以有(1,2),(1,3),(2,3)这三种选择方法.根据组合数的定 义,我们 ...