C#集合之鏈表

来源:http://www.cnblogs.com/afei-24/archive/2017/05/09/6830160.html
-Advertisement-
Play Games

LinkedList<T>是一個雙向鏈表,其元素會指向它前面和後面的元素。這樣,通過移動到下一個元素可以正向遍歷鏈表,通過移動到前一個元素可以反向遍歷鏈表。 鏈表在存儲元素時,不僅要存儲元素的值,還必須存儲每個元素的下一個元素和上一個元素的信息。這就是LinkedList<T>包含LinkedLis ...


  LinkedList<T>是一個雙向鏈表,其元素會指向它前面和後面的元素。這樣,通過移動到下一個元素可以正向遍歷鏈表,通過移動到前一個元素可以反向遍歷鏈表。
  
  鏈表在存儲元素時,不僅要存儲元素的值,還必須存儲每個元素的下一個元素和上一個元素的信息。這就是LinkedList<T>包含LinkedListNode<T>類型的元素的原因。使用LinkedListNode<T>,可以獲得列表中的下一個和上一個元素。LinkedListNode<T>定義了屬性List,Next,Previous和Value。List屬性返回與節點相關的LinkedList<T>對象。Next和Previous屬性用於遍歷鏈表,訪問當前節點之後和之前的節點。Value屬性返回與節點相關的元素,其類型是T。
  鏈表的優點是,如果將元素插入到列表的中間位置,使用鏈表就會很快。在插入一個元素時,秩序啊喲修改上一個元素的Next引用和下一個元素的Previous引用,使它們引用所插入的元素。在  List<T>(http://www.cnblogs.com/afei-24/p/6824791.html)中,插入一個元素,需要移動該元素後面的所以元素。
  鏈表的缺點是,鏈表元素只能一個接一個的訪問,這需要較長時間來查找位於鏈表中間或尾部的元素。

  LinkedList<T>類定義的成員可以訪問鏈表中的第一個和最後一個元素(First和Last);
  在指定位置插入元素:AddAfter(),AddFirst()和AddLast();
  刪除指定位置的元素:Remove(),RemoveFirst(),RemoveLast();
  搜索:Find(),FindLast()。

  下麵用一個例子演示鏈表。在鏈表中,文檔按照優先順序來排序。如果多個文檔的優先順序相同,這些元素就按照文檔的插入時間來排序:
  PriorityDocumentManager類使用一個鏈表LinkedList<Document> documentList和一個列表List<LinkedListNode<Document>> priorityNodes,鏈表包含Document對象,Document對象包含文檔的標題和優先順序。列表List<LinkedListNode<Document>> priorityNodes應最多包含10個元素,每個元素分別是引用每個優先順序的最後一個文檔對象。
  

  

      public class PriorityDocumentManager
          {
            private readonly LinkedList<Document> documentList;
               
            // priorities 0.9
            private readonly List<LinkedListNode<Document>> priorityNodes;

            public PriorityDocumentManager()
            {
              documentList = new LinkedList<Document>();

              priorityNodes = new List<LinkedListNode<Document>>(10);
              for (int i = 0; i < 10; i++)
              {
                priorityNodes.Add(new LinkedListNode<Document>(null));
              }
            }

            public void AddDocument(Document d)
            {
              //Contract.Requires<ArgumentNullException>(d != null, "argument d must not be null");
              if (d == null) throw new ArgumentNullException("d");

              AddDocumentToPriorityNode(d, d.Priority);
            }

            private void AddDocumentToPriorityNode(Document doc, int priority)
            {
                    if (priority > 9 || priority < 0)
                        throw new ArgumentException("Priority must be between 0 and 9");

              //檢查優先順序列表priorityNodes中是否有priority這個優先順序
              if (priorityNodes[priority].Value == null)
              {
                //如果沒有,遞減優先順序值,遞歸AddDocumentToPriorityNode方法,檢查是否有低一級的優先順序
                --priority;
                if (priority >= 0)
                {
                  AddDocumentToPriorityNode(doc, priority);
                }
                else //如果已經沒有更低的優先順序時,就直接在鏈表中添加該節點,並將這個節點添加到優先順序列表
                {
                  documentList.AddLast(doc);
                  priorityNodes[doc.Priority] = documentList.Last;
                }
                return;
              }
              else //優先順序列表priorityNodes中有priority這個優先順序
              {
                LinkedListNode<Document> prioNode = priorityNodes[priority];
                //區分優先順序列表priorityNodes存在這個指定的優先順序值的節點,還是存在較低的優先順序值的節點
                if (priority == doc.Priority)
                // 如果存在這個指定的優先順序值的節點
                {
                  //將這個節點添加到鏈表
                  documentList.AddAfter(prioNode, doc);

                  // 將這個節點賦予優先順序列表中的這個優先順序值的節點,因為優先順序節點總是引用指定優先順序節點的最後一個文檔
                  priorityNodes[doc.Priority] = prioNode.Next;
                }
                else //如果存在較低的優先順序值的節點
                {
                  //在鏈表中找到這個較低優先順序的第一個節點,把要添加的節點放到它前面
                  LinkedListNode<Document> firstPrioNode = prioNode;
                    //通過迴圈,使用Previous找到這個優先順序的第一個節點
                  while (firstPrioNode.Previous != null &&
                     firstPrioNode.Previous.Value.Priority == prioNode.Value.Priority)
                  {
                    firstPrioNode = prioNode.Previous;
                    prioNode = firstPrioNode;
                  }

                  documentList.AddBefore(firstPrioNode, doc);

                  // 設置一個新的優先順序節點
                  priorityNodes[doc.Priority] = firstPrioNode.Previous;
                }
              }
            }

            public void DisplayAllNodes()
            {
              foreach (Document doc in documentList)
              {
                Console.WriteLine("priority: {0}, title {1}", doc.Priority, doc.Title);
              }
            }

            // returns the document with the highest priority
            // (that's first in the linked list)
            public Document GetDocument()
            {
              Document doc = documentList.First.Value;
              documentList.RemoveFirst();
              return doc;
            }

          }

          //存儲在鏈表中的元素是Document類型
          public class Document
          {
            public string Title { get; private set; }
            public string Content { get; private set; }
            public byte Priority { get; private set; }

            public Document(string title, string content, byte priority)
            {
              this.Title = title;
              this.Content = content;
              this.Priority = priority;
            }
          }

 

客戶端代碼:
  

static void Main()
        {
          PriorityDocumentManager pdm = new PriorityDocumentManager();
          pdm.AddDocument(new Document("one", "Sample", 8));
          pdm.AddDocument(new Document("two", "Sample", 3));
          pdm.AddDocument(new Document("three", "Sample", 4));
          pdm.AddDocument(new Document("four", "Sample", 8));
          pdm.AddDocument(new Document("five", "Sample", 1));
          pdm.AddDocument(new Document("six", "Sample", 9));
          pdm.AddDocument(new Document("seven", "Sample", 1));
          pdm.AddDocument(new Document("eight", "Sample", 1));

          pdm.DisplayAllNodes();

          Console.ReadKey();

        }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 更新yum # yum update 新建用戶 # adduser user設置密碼 # passwd user 允許用戶通過ssl遠程訪問 # vi /etc/ssh/sshd_config 在文末加上 AllowUsers user1 user2 修改許可權 # vi /etc/passwd 將U ...
  • ^ 一行的開始標誌如^bigeyyes匹配到所有以bigeyyes開頭的行 $ 一行的結束標誌如$bigeyyes 匹配到所有以bigeyyes結尾的行 ?或 . 匹配任意一個非換行字元,如big?eyes匹配到big後接一個任意字元,然後是eyyes的行 * 匹配任意0個或者多個字元 [xxx]或 ...
  • 新建一個空的項目 新建好了空的項目以後,接著通過NuGet安裝一下三個包 Nancy Nancy.Hosting.Aspnet Nancy.ViewEnglines.Razor 然後在項目中添加Models,Module,Views三個文件夾,併在Models中添加UserModel類 然後往Mod ...
  • 1. 原則 推薦以符合以下原則的方式編寫模板化控制項: 選擇合適的父類: 選擇合適的父類可以節省大量的工作,從UWP自帶的控制項中選擇父類是最安全的做法,通常的選擇是Control、ContentControl、ItemsControl,也可以選擇從RangeBase、Selector中。 代碼和UI分 ...
  • 我們想在一個文本框輸入一些文字,然後點擊銨鈕,alert()出來。 <div ng-app="alertApp" ng-controller="alertController"> <div> <label>Name:</label> <input type="text" ng-model="Name ...
  • 項目需求原因需要把Webapi中的Datetime 序列化及反序列化時間戳(long),遇到相同問題的同學可作參考。 1.聲明一個時間戳轉換器 2.配置使用時間戳轉換器(到這一步API就能序列化和反序列化時間戳了) 3.因為項目中使用了Swagger UI自動生成WebApi文檔如果想介面文檔Dat ...
  • 第一種方案: 用require吧 <configuration> ... <startup> <requiredRuntime version="4.0.30319" safemode="true"/> </startup> ... </configuration> 轉載自:https://soci ...
  • 如果需要基於鍵對所需集合排序,就可以使用SortedList<TKey,TValue>類。這個類按照鍵給元素排序。這個集合中的值和鍵都可以使用任何類型。定義為鍵的自定義類型需要實現IComparer<T>介面,用於給列表中的元素排序。 使用構造函數創建一個有序列表,在用Add方法添加: var bo ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...