欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++ 漢諾塔問題知識點總結(jié)

 更新時間:2020年02月19日 14:26:44   作者:000紫外線000  
在本篇文章里小編給大家整理的是關(guān)于C++ 漢諾塔問題知識點內(nèi)容,有需要的朋友們可以參考下。

漢諾塔問題,是心理學(xué)實驗研究常用的任務(wù)之一。當(dāng)然我們是學(xué)計算機的,因此我們嘗試用計算機去求解它。

例題

openjudge6261 漢諾塔問題

描述

有一種智力玩具,在一塊銅板上有三根桿,最左邊的桿上自上而下、由小到大順序串著由n個圓盤構(gòu)成的塔。目的是將最左邊桿上的盤全部移到中間的桿上,條件是一次只能移動一個盤,且不允許大盤放在小盤的上面。這就是著名的漢諾塔問題。

假定圓盤從小到大編號為1,2,3,……

輸入

輸入為一個整數(shù)后面跟三個單字符字符串。

整數(shù)為盤子的數(shù)目,后三個字符表示三個桿子的編號。

輸出

輸出每一步移動盤子的記錄。一次移動一行。

每次移動的記錄為例如 a->3->b 的形式,即把編號為3的盤子從a桿移至b桿。

樣例輸入

2 a b c

樣例輸出

a->1->c
a->2->b
c->1->b

漢諾塔問題

漢諾塔問題的解決算法是一種經(jīng)典的分治算法,而分治算法最重要的三個步驟:

  1. 分解:如果說我們要將num個盤子從原柱子l通過過渡柱子mid移動到目標(biāo)柱子r,那么我們可以先把上面的(num - 1)個盤子從原柱子l移動到過渡柱子mid,之后再把編號num的這個盤子移動到目標(biāo)柱子r上,最后再把那(num - 1)個盤子從過渡柱子mid移動到目標(biāo)柱子r,就成功了。
  2. 解決:用遞歸分別再去解決子問題并輸出。(邊界條件:當(dāng)只有一個盤子既num == 1時,直接輸出就好了)。
  3. 合并:遞歸回來的就是結(jié)果了,不用再合并。

簡而言之,就是每次我們把第num個盤子單獨看成一個整體,剩下(num - 1)個盤子看成一個整體,之后對這兩個整體分別去進行移動,使其到達(dá)目標(biāo)位置。

最后算一下時間復(fù)雜度,這里稍微有些難算。

假設(shè)i個盤子從一根柱子移動到另一根柱子需要step(i)步

對于一個單獨的塔,程序會進行以下操作:

  1. 將上面的(n - 1)個盤子移動到過渡柱子,次數(shù)為step(n - 1)。
  2. 將第n個盤子移動到目標(biāo)柱子,次數(shù)為1。
  3. 將過渡柱子上的(n - 1)個盤子移動到目標(biāo)柱子,次數(shù)為step(n - 1)。

則可以得到遞推式

step(n) = 2 * step(n - 1) + 1

之后不停地遞推下去,就會得到

step(n) = 2^n * step(0) + 2^(n - 1) + 2^(n - 2) + ...... + 2^1 + 2^0

又因為0個盤子根本不用移,所以step(0) = 0

所以step(n) = 2^(n - 1) + 2^(n - 2) + ...... + 2^1 + 2^0

之后用等比數(shù)列的公式就可以推出:step(n) = 2^n^ - 1

我們發(fā)現(xiàn)移動次數(shù)為2^n^ - 1,實際上這也是漢諾塔問題最少的移動次數(shù)。所以最后得出解決漢諾塔問題的算法時間復(fù)雜度為O(2^n^)。

代碼

# include <cstdio>
# include <iostream> 
# include <cmath>
# include <cstring>
# include <algorithm>

using namespace std;

int n;
char a, b, c;

// hanoi(num, l, mid, r)表示需要將num個盤子從柱子l通過柱子mid移動到柱子r。
void hanoi(int num, char l, char mid, char r)
{
  if (num == 1) printf("%c->%d->%c\n", l, num, r);
  else {
    hanoi(num - 1, l, r, mid);
    printf("%c->%d->%c\n", l, num, r);
    hanoi(num - 1, mid, l, r);
  }
}

int main()
{
  scanf("%d", &n);
  cin >> a >> b >> c;
  hanoi(n, a, c, b); // 這里因為題目中是讓所有盤子從左面的柱子移動到中間的柱子,既從a到b。
  return 0;
}

就是小編整理的全部相關(guān)知識點,感謝大家的學(xué)習(xí)和對腳本之家的支持。

您可能感興趣的文章:

相關(guān)文章

  • C語言中回調(diào)函數(shù)的使用詳情

    C語言中回調(diào)函數(shù)的使用詳情

    這篇文章主要介紹了C語言中回調(diào)函數(shù)的使用詳情,閱讀下文我們將學(xué)習(xí)到架構(gòu)的核心理念和需、回調(diào)函數(shù)的作用、回調(diào)函數(shù)的程序編寫等內(nèi)容,需要的小伙伴可以參考一下
    2022-03-03
  • C/C++編寫推箱子小游戲

    C/C++編寫推箱子小游戲

    這篇文章主要為大家詳細(xì)介紹了C/C++編寫推箱子小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • C語言代碼 模塊化實現(xiàn)三子棋

    C語言代碼 模塊化實現(xiàn)三子棋

    這篇文章主要為大家詳細(xì)介紹了C語言 模塊化實現(xiàn)三子棋程序,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • C++實現(xiàn)并行計算的兩種方式

    C++實現(xiàn)并行計算的兩種方式

    本文介紹了使用C++實現(xiàn)并行計算的兩種方式,包括OpenMP和MPI,文中通過示例代碼介紹的非常詳細(xì),需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-03-03
  • C語言 動態(tài)內(nèi)存分配詳解

    C語言 動態(tài)內(nèi)存分配詳解

    這篇文章主要介紹了C語言 動態(tài)內(nèi)存分配詳解的相關(guān)資料,需要的朋友可以參考下
    2017-06-06
  • 解析C++編程中的繼承方面的運用

    解析C++編程中的繼承方面的運用

    這篇文章主要介紹了解析C++編程中的繼承方面的運用,是C++入門學(xué)習(xí)中的基礎(chǔ)知識,需要的朋友可以參考下
    2015-09-09
  • Qt實現(xiàn)打地鼠游戲的方法詳解

    Qt實現(xiàn)打地鼠游戲的方法詳解

    這篇文章主要和大家詳細(xì)介紹了如何利用Qt實現(xiàn)一個簡單的打地鼠游戲,文中的示例代碼講解詳細(xì),具有一定的借鑒價值,需要的可以參考一下
    2022-10-10
  • 淺析C++模板類型中的原樣轉(zhuǎn)發(fā)和可變參數(shù)的實現(xiàn)

    淺析C++模板類型中的原樣轉(zhuǎn)發(fā)和可變參數(shù)的實現(xiàn)

    可變參數(shù)模板(variadic templates)是C++11新增的強大的特性之一,它對模板參數(shù)進行了高度泛化,能表示0到任意個數(shù)、任意類型的參數(shù),這篇文章主要介紹了C++可變參數(shù)模板的展開方式,需要的朋友可以參考下
    2022-08-08
  • MFC擴展DLL中導(dǎo)出類和對話框的實現(xiàn)方法

    MFC擴展DLL中導(dǎo)出類和對話框的實現(xiàn)方法

    這篇文章主要介紹了MFC擴展DLL中導(dǎo)出類和對話框的實現(xiàn)方法,詳細(xì)講述了實現(xiàn)擴展DLL中導(dǎo)出類和對話框的具體步驟與方法,具有不錯的實用價值,需要的朋友可以參考下
    2014-10-10
  • C++實現(xiàn)簡單計算器功能

    C++實現(xiàn)簡單計算器功能

    這篇文章主要為大家詳細(xì)介紹了C++實現(xiàn)簡單計算器功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-05-05

最新評論