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

c# 實現(xiàn)漢諾塔游戲

 更新時間:2020年11月17日 14:57:59   作者:黑衫老腰  
這篇文章主要介紹了c# 實現(xiàn)漢諾塔游戲的示例,幫助大家更好的理解和使用c# 編程語言,感興趣的朋友可以了解下

漢諾塔游戲一旦掌握了規(guī)律,其實是有點單調(diào)和無聊的,不過卻是學習遞歸的一個絕佳例子,想當初學習老譚C的時候,就卡在這兒好長時間。

對初學編程的人來說,遞歸本身就不易理解,如果使用C語言沒有好的調(diào)試環(huán)境就更難去理解了。在這方面,VS調(diào)試輸出是真方便,一目了然。

但要想理解遞歸求解漢諾塔,還是得先玩一玩游戲本身,有點感受。另外,設(shè)計一下這個游戲也可以加深對它的理解,同時還可以對數(shù)據(jù)結(jié)構(gòu)中的棧有一個簡單的認識。

因為漢諾塔的游戲規(guī)則本身就是一個棧,只能從平臺最上面取走碟子,然后放在另一個平臺最上面,這明顯是個棧。下面是我設(shè)計的游戲界面:

游戲的玩法關(guān)鍵:奇數(shù)個盤子,第一個先移到C上;偶數(shù)個盤子,第一個先移到B上

平臺和背景容器是事先畫好的,碟子是動態(tài)生成的,使用了容器控件panel的拖動事件來實現(xiàn)拖動碟子(就是個label控件而已)的功能。

主要代碼:

using System;
using System.Collections.Generic;
using System.Drawing;
using System.Windows.Forms;
 
namespace 漢諾塔
{
    public partial class Form1 : Form
    {
        int dishNum = 3;
        int dishThick = 20;
        int count;            // 記錄步數(shù)
        Stack<int> pileA;     // 存放平臺A的數(shù)據(jù)
        Stack<int> pileB;     // 存放平臺B的數(shù)據(jù)
        Stack<int> pileC;     // 存放平臺C的數(shù)據(jù)
 
        public Form1()
        {
            InitializeComponent();
            InitGame();
            Result.ShowAnswer(dishNum);
        }
 
        private void InitGame()
        {
            pileA = new Stack<int>();
            pileB = new Stack<int>();
            pileC = new Stack<int>();
            panelA.Tag = pileA;
            panelB.Tag = pileB;
            panelC.Tag = pileC;
            for (int i = 0; i < dishNum; i++)
            {
                Label label = new Label();
                label.BackColor = Color.Gold;
                label.BorderStyle = BorderStyle.FixedSingle;
                label.Parent = panelA;
                label.Size = new Size(panelA.Width - (i + 1) * 12, dishThick);    // 從下往上每層寬度減少12
                label.Location = new Point(panelA.Width / 2 - label.Width / 2, panelA.Height - dishThick * (i + 1));
                label.BringToFront();
                label.MouseMove += new MouseEventHandler(Dish_MouseMove);
                label.Text = (dishNum - i).ToString();
                label.TextAlign = ContentAlignment.MiddleCenter;
                label.Name = "dish" + i;
                label.Tag = i;    // 設(shè)置碟子的編號
                pileA.Push(i);    // 存儲碟子的編號
            }
            count = 0;
            lblCount.Text = "已走步數(shù):" + count;
            lblLevel.Text = "最少步數(shù):" + (Math.Pow(2, dishNum) - 1);
        }
 
        private void Dish_MouseMove(object sender, MouseEventArgs e)
        {
            Label label = sender as Label;
            if (e.Button == MouseButtons.Left)
            {
                label.DoDragDrop(label, DragDropEffects.Move);
            }
        }
 
        private void panelBox_DragEnter(object sender, DragEventArgs e)
        {
            if (e.Data.GetDataPresent(typeof(Label)))
            {
                //Console.WriteLine("DragEnter: {0}", (sender as Panel).Name);
                Label label = e.Data.GetData(typeof(Label)) as Label;
                Panel panelFrom = label.Parent as Panel;
                Stack<int> pileFrom = panelFrom.Tag as Stack<int>;
                int index = (int)label.Tag;
                if (pileFrom.Peek() == index) e.Effect = DragDropEffects.Move;    // 只可以拖走最頂層的碟子
                else e.Effect = DragDropEffects.None;
            }
        }
 
        private void panelBox_DragDrop(object sender, DragEventArgs e)
        {
            Panel panelTo = sender as Panel;
            //Console.WriteLine("DragDrop: {0}", panelTo.Name);
            //Point newPoint = panel.PointToClient(new Point(e.X, e.Y));
            Stack<int> pileTo = panelTo.Tag as Stack<int>;
            Label label = e.Data.GetData(typeof(Label)) as Label;
            Panel panelFrom = label.Parent as Panel;
            Stack<int> pileFrom = panelFrom.Tag as Stack<int>;
            int index = (int)label.Tag;
            if (pileTo.Count == 0 || index > pileTo.Peek())    // 只可以放置小的碟子到大的碟子上(棧為空時使用Peek會引發(fā)異常)
            {
                label.Parent = panelTo;
                label.Location = new Point(panelTo.Width / 2 - label.Width / 2, panelTo.Height - dishThick * (pileTo.Count + 1));
                label.BringToFront();
                pileFrom.Pop();
                pileTo.Push(index);
                count++;
                lblCount.Text = "已走步數(shù):" + count;
                if (IsWin())
                {
                    if (dishNum < 9)    // 9層夠玩了
                    {
                        MessageBox.Show("OK, be going to next...", "Help", MessageBoxButtons.OK, MessageBoxIcon.Information);
                        dishNum++;
                        Reset();
                        InitGame();
                    }
                    else
                    {
                        MessageBox.Show("You Win!", "Help", MessageBoxButtons.OK, MessageBoxIcon.Information);
                    }
                }
            }
            else e.Effect = DragDropEffects.None;
        }
 
        private bool IsWin()
        {
            if (pileC.Count == dishNum) return true;
            else return false;
        }
 
        private void Reset()    // 刪除現(xiàn)在所有的碟子
        {
            //panelC.Controls.Clear();    // 會把柱子也刪除
            //Console.WriteLine("panelC: {0}", panelC.Controls.Count);
            for (int i = panelC.Controls.Count - 1; i >= 0; i--)    // 注意從前往后移除控件時控件的索引會跟著變化導(dǎo)致難以跟蹤
            {
                //Console.WriteLine(" {0} {1}", i, panelC.Controls[i].Name);
                if (panelC.Controls[i].Name.StartsWith("dish")) panelC.Controls[i].Dispose();
            }
        }
    }
}

用C#寫的代碼感覺都不用怎么解釋,大概思路一說,關(guān)鍵的地方注釋一下,一看就明白了。

然后說到解法,其實就是使用老譚C的代碼,不過標注了有用的調(diào)試輸出信息,如下:

namespace 漢諾塔
{
  class Result
  {
    static int count = 0;

    static void Move(char x, char y)         // 從x座移到y(tǒng)座
    {
      Debug.Print("  {0} ------> {1}", x, y);   // 實際執(zhí)行
      //Console.WriteLine(" {0} --> {1}", x, y);
      count++;   // 把count++放到hanoi中兩個move后面也可以
    }

    static void Hanoi(int n, char a, char b, char c)   // a:源座,b:中轉(zhuǎn)座,c:目標座;將n個盤從a座借助b座移到c座
    {
      Debug.Print("{0}: {1} -> {2} -> {3}", n, a, b, c);  // 目標任務(wù)
      if (n == 1)
      {
        Move(a, c);
      }
      else
      {
        Hanoi(n - 1, a, c, b);   // 分解任務(wù)(遞歸調(diào)用相對于將此任務(wù)作為目標任務(wù)以便繼續(xù)分解)
        Debug.Print("{0}: {1} ------> {2}", n, a, c);  // 實際任務(wù)
        Move(a, c);
        Hanoi(n - 1, b, a, c);   // 分解任務(wù)(遞歸調(diào)用相對于將此任務(wù)作為目標任務(wù)以便繼續(xù)分解)
      }
    }

    public static void ShowAnswer(int num)
    {
      Hanoi(num, 'A', 'B', 'C');
      Console.WriteLine("總共需{0}步", count);
    }
  }
}

理解1:移動n個盤子的步數(shù)是移動n-1個盤子的步數(shù)的2倍再加1,即:a[n] = 2a[n-1]+1,可推出:a[n]=2^n-1
理解2:n個盤子分配給n個人的話,第1人(總工)只需走1步,第2人(分包)走2步,第3人(分包)走4步...第n人(工人)走2^(n-1)步
理解3:從最上層開始層層分解任務(wù),下面的所有層次干完后他再干然后下層繼續(xù)補完,每個人都干活了,只有最底層的人接到任務(wù)不做分解直接開干
查看調(diào)試輸出可以很容易的理解整個調(diào)用過程,有碟子數(shù)量和雙箭頭的為目標任務(wù),有碟子數(shù)量和單箭頭的為實際任務(wù),只有單箭頭的為實際執(zhí)行步驟

3: A -> B -> C
2: A -> C -> B
1: A -> B -> C
  A ------> C
2: A ------> B
  A ------> B
1: C -> A -> B
  C ------> B
3: A ------> C
  A ------> C
2: B -> A -> C
1: B -> C -> A
  B ------> A
2: B ------> C
  B ------> C
1: A -> B -> C
  A ------> C
總共需7步

將n個盤從a座借助b座移到c座,將上面的數(shù)據(jù)代入就可以很容易的理解了。下載地址

以上就是c# 實現(xiàn)漢諾塔游戲的詳細內(nèi)容,更多關(guān)于c# 漢諾塔游戲的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • C# Assembly.Load案例詳解

    C# Assembly.Load案例詳解

    這篇文章主要介紹了C# Assembly.Load案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下
    2021-08-08
  • c#圖片添加水印的實例代碼

    c#圖片添加水印的實例代碼

    這篇文章介紹了c#圖片添加水印的實例代碼,有需要的朋友可以參考一下
    2013-07-07
  • C#簡單爬蟲案例分享

    C#簡單爬蟲案例分享

    這篇文章主要為大家分享了C#簡單爬蟲案例,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • c#集合快速排序類實現(xiàn)代碼分享

    c#集合快速排序類實現(xiàn)代碼分享

    這篇文章主要介紹了C#實現(xiàn)集合排序類,大家參考使用吧
    2013-12-12
  • 一文搞懂C# 數(shù)據(jù)類型

    一文搞懂C# 數(shù)據(jù)類型

    這篇文章主要介紹C# 數(shù)據(jù)類型的相關(guān)資料,文中講解非常細致,代碼幫助大家更好的理解和學習,感興趣的朋友可以了解下
    2020-07-07
  • c#用Treeview實現(xiàn)FolderBrowerDialog 和動態(tài)獲取系統(tǒng)圖標(運用了Win32 dll類庫)

    c#用Treeview實現(xiàn)FolderBrowerDialog 和動態(tài)獲取系統(tǒng)圖標(運用了Win32 

    其實,FolderBrowerDialog 很好用呢,有木有啊親,反正我特別的喜歡,微軟大哥把這個瀏覽文件夾的東東封裝的多好呀
    2013-03-03
  • C#使用AutoMapper進行對象映射的示例代碼

    C#使用AutoMapper進行對象映射的示例代碼

    AutoMapper 是一個對象到對象映射的庫,可以簡化 DTO (Data Transfer Objects) 和實體類之間的轉(zhuǎn)換,在大型應(yīng)用程序中,通常需要將業(yè)務(wù)實體映射到視圖模型或 DTO 中,本文將詳細介紹如何在 C# 項目中使用 AutoMapper,包括安裝、配置、以及示例代碼
    2024-08-08
  • C#實現(xiàn)繪制浮雕圖片效果實例

    C#實現(xiàn)繪制浮雕圖片效果實例

    這篇文章主要介紹了C#實現(xiàn)繪制浮雕圖片效果實例,是C#程序設(shè)計中非常實用的一個功能,需要的朋友可以參考下
    2014-08-08
  • C#在 .NET中使用依賴注入的示例詳解

    C#在 .NET中使用依賴注入的示例詳解

    這篇文章主要為大家詳細介紹了C#如何在 .NET中使用依賴注入,文中的示例代碼講解詳細,具有一定的學習價值,感興趣的小伙伴可以跟隨小編一起了解一下
    2024-01-01
  • c#通用登錄模塊分享

    c#通用登錄模塊分享

    這是一款簡單的ASP.NETC#注冊登錄模塊制作思路,非常簡單實用,雖然沒怎么考慮登陸的安全性,但作為C#的朋友學習交流使用。
    2016-07-07

最新評論