c#漢諾塔的遞歸算法與解析
從左到右 A B C 柱 大盤(pán)子在下, 小盤(pán)子在上, 借助B柱將所有盤(pán)子從A柱移動(dòng)到C柱, 期間只有一個(gè)原則: 大盤(pán)子只能在小盤(pán)子的下面.
如果有3個(gè)盤(pán)子, 大中小號(hào), 越小的越在上面, 從上面給盤(pán)子按順序編號(hào) 1(小),2(中),3(大), 后面的原理解析引用這里的編號(hào).
小時(shí)候玩過(guò)這個(gè)游戲, 基本上玩到第7個(gè),第8個(gè)就很沒(méi)有耐心玩了,并且操作的動(dòng)作都幾乎相同覺(jué)得無(wú)聊. 后來(lái)學(xué)習(xí)編程, 認(rèn)識(shí)到遞歸, 用遞歸解決漢諾塔的算法也是我除了簡(jiǎn)單的排序算法后學(xué)習(xí)到的第一種算法.
至于遞歸,簡(jiǎn)單來(lái)說(shuō)就是方法內(nèi)部自己調(diào)用自己, 同時(shí)也一定有一個(gè)結(jié)束點(diǎn). 如果對(duì)方法調(diào)用棧了解的話(huà),其實(shí)是很容易理解方法的調(diào)用過(guò)程的, 就是從主線(xiàn)程開(kāi)始調(diào)用方法進(jìn)行不停的壓棧和出棧操作. 方法的調(diào)入就是將方法壓入棧中, 方法的結(jié)束就是方法出棧的過(guò)程, 這樣保證了方法調(diào)用的順序流. 如果跟蹤遞歸的調(diào)用情況會(huì)發(fā)現(xiàn)也是如此, 到最后一定是這個(gè)方法最后從棧中彈出回到主線(xiàn)程, 并且結(jié)束.
棧的特點(diǎn):先進(jìn)后出。 比如一個(gè)方法 A 自己調(diào)用自己, 我用編號(hào)區(qū)分一下進(jìn)棧過(guò)程:
A -> A(1) -> A(2) -> A(3)
在A(3)時(shí)滿(mǎn)足某種條件得以退出, 回到 A(2), A(2)結(jié)束回到A(1), 再回到A, 出棧過(guò)程:
A(3) -> A(2) -> A(1) -> A
對(duì)于遞歸,還有一個(gè)形象的認(rèn)識(shí),就是我小時(shí)候家里有一個(gè)柜子, 柜子兩端都是玻璃, 頭伸進(jìn)柜子看一面鏡子,會(huì)看到鏡子里還有鏡子, 然后鏡子里還有鏡子, 但和遞歸的特點(diǎn)不同的是這鏡子的反射是沒(méi)有盡頭的, 只要眼睛一直能看到底的話(huà).
了解完遞歸后, 再回頭來(lái)看如何用遞歸的方式解決漢諾塔的問(wèn)題.
案例 1 - 假設(shè)只有一個(gè)盤(pán)子的時(shí)候, 盤(pán)子數(shù)量 N=1
只有一個(gè)步驟 將第1個(gè)盤(pán)子從A移動(dòng)到C, 為了對(duì)比方便我這樣來(lái)描述這個(gè)步驟:
步驟 盤(pán)子編號(hào) 從柱子移動(dòng) 移動(dòng)到柱子
1 1 A C
案例 2 - 如果有兩個(gè)盤(pán)子, 盤(pán)子數(shù)量 N = 2
步驟 盤(pán)子編號(hào) 從柱子移動(dòng) 移動(dòng)到柱子
1 1 A B
2 2 A C
3 1 B C
案例 3 - 如果有三個(gè)盤(pán)子, 盤(pán)子數(shù)量 N = 3
步驟 盤(pán)子編號(hào) 從柱子移動(dòng) 移動(dòng)到柱子
1 1 A C
2 2 A B
3 1 C B
4 3 A C
5 1 B A
6 2 B C
7 1 A C
如何找出盤(pán)子移動(dòng)的規(guī)律 ?
我們要做的最重要的一件事情就是永遠(yuǎn)要把最底下的一個(gè)盤(pán)子從 A 移動(dòng)到 C
看看上面從1個(gè)盤(pán)子的移動(dòng)到3個(gè)盤(pán)子的移動(dòng), 在移動(dòng)記錄中,當(dāng)盤(pán)子的編號(hào)和盤(pán)子數(shù)量相同的時(shí)候他們的步驟都是從A移動(dòng)到C (看加粗的部分),其它的步驟對(duì)等.
再觀察第3個(gè)案例中的第 1-3 步 和 第 5-7步
第 1-3 步 目的是從 A 移動(dòng)到 B 如果我們把 B 當(dāng)作終點(diǎn), 那么這里的第 1-3 步理解起來(lái)和 第2個(gè)案例的三個(gè)步驟完全相同, 都是通過(guò)一個(gè)柱子來(lái)移動(dòng),和第2個(gè)案例比起來(lái)在后面加括號(hào)來(lái)表示
1 1 A C ( A -> B)
2 2 A B ( A -> C)
3 1 C B ( B -> C)
總結(jié):將盤(pán)子B變成C即可.
第 5-7 步 目的是從 B 移動(dòng)到 C 如果我們把 C 當(dāng)作終點(diǎn), 那么這里的 5-7 步理解起來(lái)和上面也是一樣的, 和第2個(gè)案例的三個(gè)步驟也完全相同.和第2個(gè)案例比起來(lái)就是:
5 1 B A ( A -> B)
6 2 B C ( A- > C)
7 1 A C ( B -> C)
總結(jié): 將盤(pán)子B變成A即可
根據(jù)這個(gè)演示可以明確幾點(diǎn)規(guī)律:
1. 當(dāng)盤(pán)子只有一個(gè)的時(shí)候,只有一個(gè)動(dòng)作 從 A 移動(dòng)到 C 即結(jié)束.
2. 當(dāng)有N個(gè)盤(pán)子的時(shí)候, 中間的動(dòng)作都是從 A 移動(dòng)到 C, 那么表示最下面的第N個(gè)盤(pán)子移動(dòng)完畢
3. 中間動(dòng)作之上都可以認(rèn)為是: 從 A 移動(dòng)到 B
4. 中間動(dòng)作之下都可以認(rèn)為是: 從 B 移動(dòng)到 C
2,3,4 可以表示為
1 1 A B
2 2 A C
3 1 B C
這種結(jié)構(gòu)一直在重復(fù)進(jìn)行,C#不太熟悉,試著寫(xiě)寫(xiě),就有了以下代碼:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace DataStructure
{
class HanoiTower
{
public void MoveDisk(int DiskQuantity,string PositionA, string PositionB, string PositionC)
{
// If there's only one disk, then end.
if (DiskQuantity == 1)
{
Console.WriteLine("Move disk from position {0} to {1}.", PositionA, PositionC);
// Must return
return;
}
else
{
// Step 1 - Change B to C (A --> B)
MoveDisk(DiskQuantity - 1, PositionA,PositionC,PositionB);
// Step 2 - No changes (A --> C)
MoveDisk(1, PositionA, PositionB, PositionC);
// Step 3 - Change B to A (A --> C)
MoveDisk(DiskQuantity - 1, PositionB, PositionA, PositionC);
}
}
static void Main(string[] args)
{
HanoiTower hanoi = new HanoiTower();
Console.WriteLine("Please input Disk Quantity:");
int DiskQuantity = Convert.ToInt32(Console.ReadLine());
hanoi.MoveDisk(DiskQuantity, "A", "B", "C");
Console.ReadKey();
}
}
}
結(jié)合上面的分析,最重要的就是這里的3步交換動(dòng)作, 中間從 A到C的動(dòng)作是最底層盤(pán)子的最終操作.
// Step 1 - Change B to C (A --> B)
MoveDisk(DiskQuantity - 1, PositionA,PositionC,PositionB);
// Step 2 - No changes (A --> C)
MoveDisk(1, PositionA, PositionB, PositionC);
// Step 3 - Change B to A (A --> C)
MoveDisk(DiskQuantity - 1, PositionB, PositionA, PositionC);
至于第1個(gè)參數(shù)為什么是DiskQuantity - 1,或者1 大家再回到上面看看是不是所有的步驟都是.. 1. 1,2,1. 1,2,1,3,1,2,1 這種以盤(pán)子數(shù)對(duì)稱(chēng)的結(jié)構(gòu),而它前后都是重復(fù)1,2,1 的過(guò)程.
相關(guān)文章
C#處理猜拳問(wèn)題的簡(jiǎn)單實(shí)例(非窗體)
下面小編就為大家?guī)?lái)一篇C#處理猜拳問(wèn)題的簡(jiǎn)單實(shí)例(非窗體)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-07-07C# 封裝HtmlHelper組件:BootstrapHelper
這篇文章主要介紹了C# 封裝HtmlHelper組件之BootstrapHelper 的相關(guān)資料,需要的朋友可以參考下2016-08-08C#統(tǒng)計(jì)C、C++及C#程序代碼行數(shù)的方法
這篇文章主要介紹了C#統(tǒng)計(jì)C、C++及C#程序代碼行數(shù)的方法,較為詳細(xì)的分析了C#統(tǒng)計(jì)文本文件的原理與相關(guān)實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-08-08C#實(shí)現(xiàn)計(jì)算年齡的簡(jiǎn)單方法匯總
本文給大家分享的是C#代碼實(shí)現(xiàn)的簡(jiǎn)單實(shí)用的給出用戶(hù)的出生日期,計(jì)算出用戶(hù)的年齡的代碼,另外附上其他網(wǎng)友的方法,算是對(duì)計(jì)算年齡的一次小結(jié),希望大家能夠喜歡。2015-05-05C#自定義簡(jiǎn)化cookie類(lèi)實(shí)例
這篇文章主要介紹了C#自定義簡(jiǎn)化cookie類(lèi),實(shí)例分析了C#操作cookie的添加、獲取及刪除等操作的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03基于Aforge攝像頭調(diào)用簡(jiǎn)單實(shí)例
這篇文章主要為大家詳細(xì)介紹了基于Aforge攝像頭調(diào)用的簡(jiǎn)單實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-10-10C#如何實(shí)現(xiàn)調(diào)取釘釘考勤接口的功能
這篇文章主要介紹了C#如何實(shí)現(xiàn)調(diào)取釘釘考勤接口的功能,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-08-08C#之關(guān)于Base64簡(jiǎn)單加密與解密方式
這篇文章主要介紹了C#之關(guān)于Base64簡(jiǎn)單加密與解密方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06c#使用簡(jiǎn)單工廠模式實(shí)現(xiàn)生成html文件的封裝類(lèi)分享
這篇文章主要介紹了運(yùn)用了簡(jiǎn)單工廠模式實(shí)現(xiàn)頁(yè)面靜態(tài)化封裝類(lèi),思路比較簡(jiǎn)單,大家可根據(jù)自己的思路再擴(kuò)展此類(lèi)2014-01-01