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

C#數(shù)據(jù)結(jié)構(gòu)之隊列(Quene)實例詳解

 更新時間:2015年11月27日 14:55:06   作者:Jimmy.Yang  
這篇文章主要介紹了C#數(shù)據(jù)結(jié)構(gòu)之隊列(Quene),結(jié)合實例形式較為詳細的講述了隊列的功能、原理與C#實現(xiàn)隊列的相關(guān)技巧,需要的朋友可以參考下

本文實例講述了C#數(shù)據(jù)結(jié)構(gòu)之隊列(Quene)。分享給大家供大家參考,具體如下:

隊列(Quene)的特征就是“先進先出”,隊列把所有操作限制在"只能在線性結(jié)構(gòu)的兩端"進行,更具體一點:添加元素必須在線性表尾部進行,而刪除元素只能在線性表頭部進行。

先抽象接口IQuene<T>

namespace 棧與隊列
{
  public interface IQuene<T>
  {
    /// <summary>
    /// 取得隊列實際元素的個數(shù)
    /// </summary>
    /// <returns></returns>
    public int Count();
    /// <summary>
    /// 判斷隊列是否為空
    /// </summary>
    /// <returns></returns>
    public bool IsEmpty();
    /// <summary>
    /// 清空隊列
    /// </summary>
    public void Clear();
    /// <summary>
    /// 入隊(即向隊列尾部添加一個元素)
    /// </summary>
    /// <param name="item"></param>
    public void Enquene(T item);
    /// <summary>
    /// 出隊(即從隊列頭部刪除一個元素)
    /// </summary>
    /// <returns></returns>
    public T Dequene();
    /// <summary>
    /// 取得隊列頭部第一元素
    /// </summary>
    /// <returns></returns>
    public T Peek();
  }
}

下面是基于數(shù)組實現(xiàn)的示意圖:

實現(xiàn)思路:用一個數(shù)組存放所有元素,同時設(shè)置二個關(guān)鍵變量front與rear用于記錄隊列“頭”與“尾”的元素下標,當有元素入列時rear加1,當有元素出隊時front+1,而rear-front即為隊列實際元素的總數(shù).

但有一種“隊列偽滿”的特殊情況要注意,如下圖:

這張圖上面的部分:假設(shè)經(jīng)過入隊、出隊一番折騰后,rear已經(jīng)指向數(shù)組的下標最大值,而front指向在中間(即front之間的元素已經(jīng)出隊不用考慮了,相當于front下標前面的內(nèi)存區(qū)域空閑),如果這時再有一個元素入列,rear+1就超出數(shù)組下標的最大值了,但是從圖上一眼就能看出,實際上front前面還空著一堆位置可以重復(fù)利用,隊列并非真正的“滿”--這種情況稱為偽滿,為了解決這個問題,我們可以把數(shù)組想象為首尾相接的循環(huán)結(jié)構(gòu),即圖中下面部分,這時候可以讓rear重新指向到0,以便重復(fù)利用空閑的位置。

所以:入列時rear++的操作,應(yīng)該稍做修正,當rear到數(shù)組下標最大值時,讓它置0,以便能循環(huán)利用 (見后面的代碼)

另外還有一個問題:最開始時front與rear都為-1,即front==rear時表示隊列為空,改成循環(huán)以后,有可能會出現(xiàn)rear在循環(huán)過程中碰到front的情況,即真正意義的上"滿"狀態(tài),這時rear也同樣等于front,這樣就無法單純的用rear==front來判斷是滿,還是空?這時可以浪費一個元素的位置,認為當rear+1==front時,隊列就已經(jīng)滿了,雖然犧牲了一個元素的空間,但卻換來了邏輯的正確性,還是值得的。

完整實現(xiàn)如下:

using System;
using System.Text;
namespace 棧與隊列
{
  /// <summary>
  /// 循環(huán)順序隊列
  /// </summary>
  /// <typeparam name="T"></typeparam>
  public class CSeqQueue<T>:IQueue<T>
  {
    private int maxsize;
    private T[] data;
    private int front;
    private int rear;    
    public CSeqQueue(int size) 
    {
      data = new T[size];
      maxsize = size;
      front = rear = -1;
    }
    public int Count()     
    {
      if (rear > front)
      {
        return rear - front;
      }
      else
      {
        return (rear - front + maxsize) % maxsize;
      }
    }
    public void Clear() 
    {
      front = rear = -1;
    }
    public bool IsEmpty() 
    {
      return front == rear;      
    }
    public bool IsFull() 
    {      
      if (front != -1) //如果已經(jīng)有元素出隊過
      {
        return (rear + 1) % maxsize == front;//為了區(qū)分與IsEmpty的區(qū)別,有元素出隊過以后,就只有浪費一個位置來保持邏輯正確性.
      }
      else
      {
        return rear == maxsize - 1;
      }      
    }
    public void Enqueue(T item) 
    {
      if (IsFull()) 
      {
        Console.WriteLine("Queue is full");
        return;
      }
      if (rear == maxsize - 1) //如果rear到頭了,則循環(huán)重來(即解決偽滿問題)
      {
        rear = 0;
      }
      else
      {
        rear++;
      }
      data[rear] = item;
    }
    public T Dequeue() 
    {      
      if (IsEmpty()) 
      {
        Console.WriteLine("Queue is empty");
        return default(T);
      }
      if (front == maxsize - 1) //如果front到頭了,則重新置0
      {
        front = 0;
      }
      else
      {
        front++;
      }      
      return data[front];
    }
    public T Peek() 
    {
      if (IsEmpty()) 
      {
        Console.WriteLine("Queue is empty!");
        return default(T);
      }
      return data[(front + 1) % maxsize];      
    }
    public override string ToString()
    {
      if (IsEmpty()) { return "queue is empty."; }
      StringBuilder sb = new StringBuilder();
      if (rear > front)
      {
        for (int i = front + 1; i <= rear; i++)
        {
          sb.Append(this.data[i].ToString() + ",");
        }
      }
      else
      {
        for (int i = front + 1; i < maxsize; i++)
        {
          sb.Append(this.data[i].ToString() + ",");
        }
        for (int i = 0; i <= rear; i++)
        {
          sb.Append(this.data[i].ToString() + ",");
        }
      }
      return "front = " + this.front + " \t rear = " + this.rear + "\t count = " + this.Count() + "\t data = " + sb.ToString().Trim(',');
    }
  }
}

測試代碼片段:

CSeqQueue<int> queue = new CSeqQueue<int>(5);
queue.Enqueue(1);
queue.Enqueue(2);
queue.Enqueue(3);
queue.Enqueue(4);
Console.WriteLine(queue);//front = -1    rear = 3    count = 4    data = 1,2,3,4
queue.Dequeue();
Console.WriteLine(queue);//front = 0    rear = 3    count = 3    data = 2,3,4
queue.Dequeue();
Console.WriteLine(queue);//front = 1    rear = 3    count = 2    data = 3,4
queue.Enqueue(5);
Console.WriteLine(queue);//front = 1    rear = 4    count = 3    data = 3,4,5
queue.Enqueue(6);
Console.WriteLine(queue);//front = 1    rear = 0    count = 4    data = 3,4,5,6
queue.Enqueue(7);    //Queue is full
Console.WriteLine(queue);//front = 1    rear = 0    count = 4    data = 3,4,5,6
queue.Dequeue();
queue.Enqueue(7);
Console.WriteLine(queue);//front = 2    rear = 1    count = 4    data = 4,5,6,7
queue.Clear();
Console.WriteLine(queue);//queue is empty.
queue.Enqueue(1);
queue.Enqueue(2);
queue.Enqueue(3);
queue.Enqueue(4);
Console.WriteLine(queue);//front = -1    rear = 3    count = 4    data = 1,2,3,4
queue.Enqueue(5);
Console.WriteLine(queue);//front = -1    rear = 4    count = 5    data = 1,2,3,4,5
queue.Enqueue(6);    //Queue is full
Console.WriteLine(queue);//front = -1    rear = 4    count = 5    data = 1,2,3,4,5
queue.Dequeue();
queue.Dequeue();
queue.Dequeue();
queue.Dequeue();
Console.WriteLine(queue);//front = 3    rear = 4    count = 1    data = 5
queue.Dequeue();
Console.WriteLine(queue);//queue is empty.
queue.Enqueue(0);
queue.Enqueue(1);
queue.Enqueue(2);
queue.Enqueue(3);
queue.Enqueue(4);    //Queue is full
Console.WriteLine(queue);//front = 4    rear = 3    count = 4    data = 0,1,2,3
Console.WriteLine(queue.Peek());//0
queue.Dequeue();
Console.WriteLine(queue);//front = 0    rear = 3    count = 3    data = 1,2,3
queue.Dequeue();
Console.WriteLine(queue);//front = 1    rear = 3    count = 2    data = 2,3
queue.Dequeue();
Console.WriteLine(queue);//front = 2    rear = 3    count = 1    data = 3
queue.Dequeue();
Console.WriteLine(queue);//queue is empty.
queue.Enqueue(9);
Console.WriteLine(queue);//front = 3    rear = 4    count = 1    data = 9
Console.ReadLine();

當然,隊列也可以用鏈表來實現(xiàn),相對要容易很多。

先定義鏈表中的節(jié)點Node.cs

namespace 棧與隊列
{
  public class Node<T>
  {
    private T data;
    private Node<T> next;
    public Node(T data, Node<T> next) 
    {
      this.data = data;
      this.next = next;
    }
    public Node(Node<T> next) 
    {
      this.next = next;
      this.data = default(T);
    }
    public Node(T data) 
    {
      this.data = data;
      this.next = null;
    }
    public Node() 
    {
      this.data = default(T);
      this.next = null;
    }
    public T Data {
      get { return this.data; }
      set { this.data = value; }
    }
    public Node<T> Next 
    {
      get { return next; }
      set { next = value; }
    }
  }
}

為了方便,定義了很多構(gòu)造函數(shù)的重載版本,當然這些只是浮云,重點是理解結(jié)構(gòu):data用來保存數(shù)據(jù),next指出下一個節(jié)點是誰
鏈式隊列的完整實現(xiàn)LinkQueue.cs

using System;
using System.Text;
namespace 棧與隊列
{
  public class LinkQueue:IQueue
  {
    private Node front;//隊列頭
    private Node rear;//隊列尾
    private int num;//隊列元素個數(shù)
    /// 
    /// 構(gòu)造器
    /// 
    public LinkQueue() 
    {
      //初始時front,rear置為null,num置0
      front = rear = null;
      num = 0;
    }
    public int Count() 
    {
      return this.num;
    }
    public void Clear() 
    {
      front = rear = null;
      num = 0;
    }
    public bool IsEmpty() 
    {
      return (front == rear && num == 0);
    }
    //入隊
    public void Enqueue(T item) 
    {
      Node q = new Node(item);
      if (rear == null)//第一個元素入列時
      {
        front = rear = q;
      }
      else
      {
        //把新元素掛到鏈尾
        rear.Next = q;
        //修正rear指向為最后一個元素
        rear = q;
      }
      //元素總數(shù)+1
      num++;
    }
    //出隊
    public T Dequeue() 
    {
      if (IsEmpty()) 
      {
        Console.WriteLine("Queue is empty!");
        return default(T);
      }
      //取鏈首元素
      Node p = front;
      //鏈頭指向后移一位
      front = front.Next;
      //如果此時鏈表為空,則同步修正rear
      if (front == null) 
      {
        rear = null;
      }
      num--;//個數(shù)-1
      return p.Data;
    }
    public T Peek() 
    {
      if (IsEmpty()) 
      {
        Console.WriteLine("Queue is empty!");
        return default(T);
      }
      return front.Data;
    }
    public override string ToString()
    {
      if (IsEmpty()) {
        Console.WriteLine("Queue is empty!");
      }
      StringBuilder sb = new StringBuilder();
      Node node = front;
      sb.Append(node.Data.ToString());
      while (node.Next!=null)
      {
        sb.Append("," + node.Next.Data.ToString());
        node = node.Next;
      }
      return sb.ToString().Trim(',');
    }
  }
}

希望本文所述對大家C#程序設(shè)計有所幫助。

相關(guān)文章

  • C#中winform實現(xiàn)自動觸發(fā)鼠標、鍵盤事件的方法

    C#中winform實現(xiàn)自動觸發(fā)鼠標、鍵盤事件的方法

    這篇文章主要介紹了C#中winform實現(xiàn)自動觸發(fā)鼠標、鍵盤事件的方法,是C#程序設(shè)計中非常實用的功能,需要的朋友可以參考下
    2014-08-08
  • C#泛型運作原理的深入理解

    C#泛型運作原理的深入理解

    這篇文章主要給大家介紹了關(guān)于C#泛型運作原理的深入理解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • C#編寫的生辰八字計算程序

    C#編寫的生辰八字計算程序

    這篇文章主要介紹了C#編寫的生辰八字計算程序,假設(shè)一個人的公歷出生時間,范圍必須要在2012-2015年之間,因為本示例程序只提供了這幾年的農(nóng)歷數(shù)據(jù),小伙伴們參考下,可以自由擴展
    2015-03-03
  • 用 C# 編寫一個停放在任務(wù)欄上的圖標程序

    用 C# 編寫一個停放在任務(wù)欄上的圖標程序

    用 C# 編寫一個停放在任務(wù)欄上的圖標程序...
    2007-03-03
  • Unity實現(xiàn)打磚塊游戲

    Unity實現(xiàn)打磚塊游戲

    這篇文章主要為大家詳細介紹了Unity實現(xiàn)打磚塊游戲,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • c#讀取文件詳談

    c#讀取文件詳談

    你平時是怎么讀取文件的?使用流讀取。是的沒錯,C#給我們提供了非常強大的類庫(又一次吹捧了.NET一番)
    2013-09-09
  • C#簡單讀取、改變文件的創(chuàng)建、修改及訪問時間的方法

    C#簡單讀取、改變文件的創(chuàng)建、修改及訪問時間的方法

    這篇文章主要介紹了C#簡單讀取、改變文件的創(chuàng)建、修改及訪問時間的方法,涉及C#文件類SetCreationTime、SetLastWriteTime及SetLastAccessTime的相關(guān)使用技巧,需要的朋友可以參考下
    2015-07-07
  • C#繪制橢圓的方法

    C#繪制橢圓的方法

    這篇文章主要介紹了C#繪制橢圓的方法,涉及C#圖形繪制的相關(guān)技巧,需要的朋友可以參考下
    2015-06-06
  • C#中緩存的基本使用方法

    C#中緩存的基本使用方法

    項目開發(fā)過程中緩存的應(yīng)用到處可見,下面這篇文章主要給大家介紹了關(guān)于C#中緩存的基本使用方法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2018-09-09
  • C#編程自學(xué)之數(shù)據(jù)類型和變量一

    C#編程自學(xué)之數(shù)據(jù)類型和變量一

    本節(jié)課我們將學(xué)習(xí)C#編程語言的數(shù)據(jù)類型,數(shù)據(jù)類型可以分為值類型和引用類型,接著介紹變量的使用方法和作用域等內(nèi)容,為了方便大家理解,我們還會舉一些小例子作為說明。
    2015-10-10

最新評論