【讀書筆記】C#高級編程 第十章 集合

来源:http://www.cnblogs.com/dlxh/archive/2017/04/03/6661965.html
-Advertisement-
Play Games

(一)概述 數組的大小是固定的。如果元素個數是動態的,就應使用集合類。 List<T>是與數組相當的集合類。還有其它類型的集合:隊列、棧、鏈表、字典和集。 (二)列表 1、創建列表 調用預設的構造函數,就可以創建列表對象。在泛型類List<T>中,必須為聲明為列表的值指定類型。使用預設構造函數創建一 ...


(一)概述

數組的大小是固定的。如果元素個數是動態的,就應使用集合類。

List<T>是與數組相當的集合類。還有其它類型的集合:隊列、棧、鏈表、字典和集。

 

 

(二)列表

1、創建列表

調用預設的構造函數,就可以創建列表對象。在泛型類List<T>中,必須為聲明為列表的值指定類型。使用預設構造函數創建一個空列表。元素添加到列表後,列表的容量會擴大,每次添加元素達到容量上限後,容量將重新設置為原來的2倍。

例子:

List<string> strList = new List<string>();

如果列表的容量改變了,整個集合就要重新分配到一個新的記憶體塊中。使用Capacity屬性可以獲取和設置集合的容量。

例子:

strList.Capacity = 100;

集合中的元素個數可以用Count屬性讀取。

例子:

var count= strList.Count;

可以調用TrimExcess()方法去除不需要的容量。

例子:

strList.TrimExcess();

但要註意的是,元素個數超過容量的90%,TrimExcess()方法就什麼都不做。

 

(1)集合初始值設定項

還可以使用集合初始值設定項給集合賦值。

例子:

List<string> strList = new List<string>() { "zhangsan", "lisi" };

 

(2)添加元素

使用Add()方法可以給列表添加元素。

例子:

strList.Add("zhangsan");

使用AddAange()方法,可以一次性給集合添加多個元素。

例子:

strList.AddRange(new string[] { "lisi", "wangmazi" });
strList.AddRange(new List<string> { "zhaoliu", "qianqi" });

 

(3)插入元素

使用Insert()方法可以在指定位置插入元素。

例子:

strList.Insert(0, "begin");

使用InsertRange()提供了插入大量元素的功能。

例子:

strList.InsertRange(1, new string[] { "zhang", "li", "wang" });

 

(4)訪問元素

實現了IList<T>和IList介面的所有類都提供了一個索引器,可以使用索引器,通過傳遞元素號來訪問元素。

例子:

var str0 = strList[0];

處理使用foreach語句進行遍歷以外,List<T>類還提供了了ForEach()方法,該方法返回void,參數為Action<T>。ForEach()方法遍歷集合中的每一項,調用作為每一項的參數傳遞的方法。

例子:

List<string> strList = new List<string>();
strList.Add("zhangsan");
strList.AddRange(new string[] { "lisi", "wangmazi" });
strList.AddRange(new List<string> { "zhaoliu", "qianqi" });
strList.ForEach(Console.WriteLine);
Console.ReadKey();

運行以上代碼,結果如下:

 

Console.WriteLine()方法作為參數傳遞給ForEach()方法。

 

(5)刪除元素

刪除元素時,可以利用索引,也可以傳遞要刪除的元素。

例子:

strList.RemoveAt(1);

按索引刪除比較快。

RemoveRange()方法可以從集合中刪除許多元素。

strList.RemoveRange(0, 1);

還可以使用RemoveAll()方法刪除有指定特定的所有元素,還可以使用ICollection<T>介面定義的Clear()方法。

 

(6)搜索

可以使用IndexOf()、LastIndexOf()、FindIndex()、FindLastIndex()、Find()、FindLast()。如果只檢查元素是否存在,List<T>類提供了Exists()方法。

例子:

int index = strList.IndexOf("lisi");

 

(7)排序

List<T>類可以使用Sort()方法對元素排序。Sort()方法使用快速排序演算法,比較所有的元素,值到整個列表排好序位為止。

 

(8)類型轉換

使用List<T>類的ConvertAll<TOutput>()方法,可以把所有類型的集合轉換為另一種類型。

例子:

List<object> objList = strList.ConvertAll<object>(s => s);

 

 

2、只讀集合

List<T>集合中的AsReadOnly()方法返回ReadOnlyCollection<T>類型的對象。

 

 

(三)隊列

隊列(Queue<T>)是已先進先出(FIFO)的方式來處理的集合。隊列只允許在隊列中添加元素,該元素會放在隊列的末尾(Enqueue()方法),從隊列頭部獲取元素(使用Dequeue()方法)。

例子:

Queue<string> qu = new Queue<string>();
qu.Enqueue("zhangsan");
qu.Enqueue("lisi");
qu.Enqueue("wangmazi");
var dqu = qu.Dequeue();
Console.WriteLine(dqu);

運行以上代碼,結果如下:

 

需要註意的是Dequeue()方法獲取元素後會從隊列中刪除該元素,使用Peek()方法則僅獲取不刪除。使用枚舉器不會改變元素的狀態。因為Queue<T>類沒有實現IList<T>介面,所以不能用索引器訪問隊列。

 

 

(四)棧

棧(Stack<T>)與隊列(Queue<T>)很相似,只是最後添加到棧中的元素最先讀取。棧是一個後進先出(LIFO)的容器。棧用Push()方法添加元素,用Pop()方法獲取最近的元素。

例子:

Stack<string> st = new Stack<string>();
st.Push("zhangsan");
st.Push("lisi");
st.Push("wangmazi");
var pst = st.Pop();
Console.WriteLine(pst);

運行以上代碼,結果如下:

 

需要註意的是Pop()方法獲取元素後會從隊列中刪除該元素,使用Peek()方法則僅獲取不刪除。使用枚舉器不會改變元素的狀態。

 

 

(五)鏈表

LinkedList<T>是一個雙向鏈表,其元素指向它前面和後面的元素。鏈表的優點是,插入列表中間位置會很快。缺點是只能一個一個訪問。鏈表不能在列表中僅存儲值,其還會包含上一個和下一個元素的信息。

LinkedList<T>類定義的成員可以訪問鏈表中的第一個和最後一個元素(First和Last)、在指定位置插入元素(AddAfter()、AddBefore()、AddFirst()和AddLast()方法),刪除指定位置元素(Remove()、RemoveFirst()和RemoveLast()方法)、從鏈表的開頭(Find()方法)或結尾(FindLast()方法)開始搜索元素。

例子:

LinkedList<int> list = new LinkedList<int>();
list.AddFirst(2);
list.AddFirst(1);
list.AddLast(5);
list.AddLast(6);
list.AddAfter(list.Find(2),3);
list.AddBefore(list.Find(5),4);
foreach (var item in list)
{
    Console.WriteLine(item);
}
list.RemoveLast();
Console.WriteLine("移除末尾元素後:");
foreach (var item in list)
{
    Console.WriteLine(item);
}

運行以上代碼,結果如下:

 

 

 

(六)有序列表

如果需要基於鍵對所需集合排序,就可以使用SortedList<TKey,TValue>類。這個類按照鍵給元素排序。

例子:

SortedList<string, string> list = new SortedList<string, string>();
list.Add("b", "lisi");
list.Add("a", "zhangsan");
list.Add("c","wangmazi");
foreach (var item in list)
{
    Console.WriteLine("排序位置{0},是{1}",item.Key,item.Value);
}

運行以上代碼,結果如下:

 

 

 

(七)字典

字典也成為映射或散射列表。字典的主要特性是能根據鍵快速查找值。也可以自由添加和刪除元素,這有點像List<T>類,但沒有在記憶體中移動後續元素的性能開銷。

 

1、鍵的類型

用作字典的鍵必須重寫Object類的GetHashCode()方法,並實現IEquatable<T>.Equals()方法,或重寫Object類的Equals()方法。也可以實現IEqualityComparer<T>介面的GetHashCode()方法和Equals()方法。

需要註意的是,如果A.Equals(B)方法返回true,則A.GetHashCode()B.GetHashCode()方法必須總是返回相同的散列代碼。

字元串比Int32更適合用作鍵。

 

2、字典示例

Dictionary<string, string> dic = new Dictionary<string, string>();
dic.Add("a", "zhang");
dic.Add("b", "li");
dic.Add("c", "wang");
foreach (var item in dic)
{
    Console.WriteLine(item);
}

 

3、Lookup類

Lookup<TKey,TElement>類非常類似於Dictionary<TKey,TValue>類,但把鍵映射到一個值集上。Lookup<TKey,TValue>類不能像一般的字典那樣創建,而必須調用ToLookup()方法,該方法返回一個Lookup<TKey,TElement>對象。調用ToLookup()方法需要傳遞一個Func<TSource,TKey>類型的委托來定義鍵的選擇器。

例子:

var personList = new List<Person>();
personList.Add(new Person() { Name = "zhangsan", Age = 13 });
personList.Add(new Person() { Name = "lisi", Age = 15 });
personList.Add(new Person() { Name = "wangmazi", Age = 13 });
personList.Add(new Person() { Name = "zhaoliu", Age = 18 });
var personLookup = personList.ToLookup(p => p.Age);
foreach (var item in personLookup[13])
{
    Console.WriteLine(item.Name);
}

 

4、有序字典

SortedDictionary<TKey,TValue>類是一個二叉搜索樹,其中的元素根據鍵來排序。該鍵類型必須實現IComparable<TKey>介面,在沒有實現的情況下,可以傳遞一個實現了IComparer<TKey>介面的比較器。

  • SortedList<TKey,TValue>類使用的記憶體比SortedDictionary<TKey,TValue>類少。
  • SortedDictionary<TKey,TValue>類的元素插入和刪除操作比較快。
  • 在用已排好序的數據填充集合時,若不需要修改容量,SortedList<TKey,TValue>類就比較快。

 

 

(八)集

包含不重覆的集合成為“集(set)”。.NET Framework包含兩個集(HashSet<T>和SortedSet<T>),它們都實現ISet<T>介面。HashSet<T>集包含不重覆元素的無序列表,SortedSet<T>包含不重覆的有序列表。

例子:

 1 var firstNameSet1 = new HashSet<string>();
 2 firstNameSet1.Add("zhang");
 3 firstNameSet1.Add("zhang");
 4 firstNameSet1.Add("li");
 5 firstNameSet1.Add("wang");
 6 firstNameSet1.Add("zhao");
 7 foreach (var item in firstNameSet1)
 8 {
 9     Console.WriteLine(item);
10 }
11 var firstNameSet2 = new HashSet<string>();
12 firstNameSet2.Add("Smith");
13 firstNameSet2.Add("Johnson");
14 firstNameSet2.Add("Williams");
15 firstNameSet2.Add("Jones");
16 var firstNameSet = new SortedSet<string>();
17 firstNameSet.UnionWith(firstNameSet1);
18 firstNameSet.UnionWith(firstNameSet2);
19 Console.WriteLine("將兩個無序集合併到有序集:");
20 foreach (var item in firstNameSet)
21 {
22     Console.WriteLine(item);
23 }

運行以上代碼,結果如下:

 

 

 

(九)可觀察的集合

如果需要集合中的元素何時刪除或添加的信息,就可以使用ObservableCollection<T>類。

例子:

 1 static void Main(string[] args)
 2 {
 3     ObservableCollection<string> obs = new ObservableCollection<string>();
 4     obs.CollectionChanged += Obs_changeInfo;
 5     obs.Add("1");
 6     obs.Add("2");
 7     Console.ReadKey();
 8 }
 9 
10  
11 private static void Obs_changeInfo(object sender, NotifyCollectionChangedEventArgs e)
12 {
13     Console.WriteLine("觸發事件操作方法:{0}",e.Action);
14     var targetElement = e.NewItems == null ? e.OldItems : e.NewItems;
15     foreach (var item in targetElement)
16     {
17         Console.WriteLine("觸發元素:{0}", item);
18     }
19 }

運行以上代碼,結果如下:

 

 

 

(十)位數組

如果需要處理的數字有許多位,就可以使用BitArray類和BitVector32結構。(完全不知道這個有什麼作用,暫不做筆記)

 

 

(十一)不變的集合

如果對象可以改變其狀態,就很難在多個同時運行的任務中使用。先填充集合,再將它變為不變的數組會更高效。需要進行一些處理時,可以再次使用可變的集合。此時可以使用不變集合提供的構建器類。

例子:

List<string> list = new List<string>();
list.Add("zhang");
list.Add("li");
var immutableList = list.ToImmutableList();//可變集合轉為不變集合
var immutableListBuilder = immutableList.ToBuilder();//重新轉換為可變集合
immutableListBuilder.Add("wang");
immutableList = immutableListBuilder.ToImmutable();//可變集合轉為不變集合
foreach (var item in immutableList)
{
    Console.WriteLine(item);
}

運行以上代碼,結果如下:

 

只讀集合提供集合的只讀試圖。在不使用只讀試圖訪問集合的情況下,該集合仍可修改。而不變集合無法修改。

 

 

(十二)併發集合

從.NET 4開始,.NET中名稱空間System.Collections.Concurrent中提供了幾個線程安全的集合類。為了對集合進行線程安全的訪問,定義了IProducerConsumerCollection<T>介面。這個介面中最重要的方法是TryAdd()方法和TryTake()方法。

併發集合參考:http://blog.csdn.net/wangzhiyu1980/article/details/45497907


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

-Advertisement-
Play Games
更多相關文章
  • Epplus操作Excel基礎詳解 1.什麼是Epplus Epplus是一個使用Open Office XML文件格式,能讀寫Excel2007/2010文件的開源組件,在導出Excel的時候不需要電腦上安裝office。 其中,Open Office XML文檔格式包括我們常見的xlsx、doc ...
  • 一步步打造一個簡單的 MVC 電商網站 - BooksStore(四) 本系列的 GitHub地址:https://github.com/liqingwen2015/Wen.BooksStore 《一步步打造一個簡單的 MVC 電商網站 - BooksStore(一)》(發佈時間:2017-03-3 ...
  • 開發工具:VS2010,MVC4.0,SQLSERVER2008 伺服器:Windows server 2012,IIS8,SQLSERVER2012 一、發佈後,每個頁面第一次打開都很卡,50秒或更長,第二次打開就很快了 估計原因:編譯速度慢,但在有VS環境的機器上發佈沒有這麼卡 解決辦法: 使用... ...
  • ElasticSearch NEST Client 操作Index var indexName="twitter"; var deleteIndexResponse = client.DeleteIndex(indexName); var createIndexResponse = client.C ...
  • 值類型:int double char bool decimal struct enum 值存儲在記憶體的棧上 引用類型: string 數組 自定義類 介面 委托 值存儲在堆中 值傳遞:把值類型作為參數傳遞,傳遞的是值本身 註:ref可以把值傳遞改變為引用傳遞 引用傳遞:把引用類型的值作為參數傳遞, ...
  • 設計模式(Design pattern):指的是一種大多數人反覆使用的代碼設計經驗。 作用:代碼復用、易讀性、保證代碼可靠性。 一、簡單工廠模式 用法:創建一個工廠類(命名習慣以Factiory結尾),一個靜態的帶參數(根據不同的參數返回不同的子類對象)的父類類型的方法。 public class ...
  • 記錄點點滴滴知識,為了更好的服務後來者! 一、為什麼使用AutoFac? 之前介紹了Unity和Ninject兩個IOC容器,但是發現園子里用AutoFac的貌似更為普遍,於是捯飭了兩天,發現這個東東確實是個高大上的IOC容器~ Autofac是.NET領域最為流行的IOC框架之一,傳說是速度最快的 ...
  • 1.看到標題首先要想到三層架構是什麼? 三層架構其實為: ①表示層:負責接收用戶的輸入,將輸出呈現給用戶, 以及訪問安全性驗證,並對輸入的數據的正確性、 有效性及呈現樣式負責,但對輸出的數據的正確性 不負責。 ②業務邏輯性:負責系統領域業務的處理,負責邏輯性數據的生產、 處理及轉換。對所輸入的邏輯性 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...