C#中HashTable虹钮、Dictionary聋庵、ConcurrentDictionar

C#中HashTable、Dictionary芙粱、ConcurrentDictionar三者都表示鍵/值對(duì)的集合祭玉,但是到底有什么區(qū)別,下面詳細(xì)介紹

一春畔、HashTable

HashTable表示鍵/值對(duì)的集合脱货。在.NET Framework中,Hashtable是System.Collections命名空間提供的一個(gè)容器律姨,用于處理和表現(xiàn)類(lèi)似key-value的鍵值對(duì)振峻,其中key通常可用來(lái)快速查找择份,同時(shí)key是區(qū)分大小寫(xiě)扣孟;value用于存儲(chǔ)對(duì)應(yīng)于key的值。Hashtable中key-value鍵值對(duì)均為object類(lèi)型荣赶,所以Hashtable可以支持任何類(lèi)型的keyvalue鍵值對(duì)凤价,任何非 null 對(duì)象都可以用作鍵或值。

HashTable是一種散列表讯壶,他內(nèi)部維護(hù)很多對(duì)Key-Value鍵值對(duì)料仗,其還有一個(gè)類(lèi)似索引的值叫做散列值(HashCode),它是根據(jù)GetHashCode方法對(duì)Key通過(guò)一定算法獲取得到的伏蚊,所有的查找操作定位操作都是基于散列值來(lái)實(shí)現(xiàn)找到對(duì)應(yīng)的Key和Value值的立轧。

散列函數(shù)(GetHashCode)讓散列值對(duì)應(yīng)HashTable的空間地址盡量不重復(fù)。

當(dāng)一個(gè)HashTable被占用一大半的時(shí)候我們通過(guò)計(jì)算散列值取得的地址值可能會(huì)重復(fù)指向同一地址躏吊,這就造成哈希沖突氛改。

C#中鍵值對(duì)在HashTable中的位置Position= (HashCode& 0x7FFFFFFF) % HashTable.Length,C#是通過(guò)探測(cè)法解決哈希沖突的比伏,當(dāng)通過(guò)散列值取得的位置Postion以及被占用的時(shí)候胜卤,就會(huì)增加一個(gè)位移x值判斷下一個(gè)位置Postion+x是否被占用,如果仍然被占用就繼續(xù)往下位移x判斷Position+2*x位置是否被占用赁项,如果沒(méi)有被占用則將值放入其中葛躏。當(dāng)HashTable中的可用空間越來(lái)越小時(shí)快骗,則獲取得到可用空間的難度越來(lái)越大郭毕,消耗的時(shí)間就越多。

使用方法如下:

復(fù)制代碼

<pre style="margin-top:0px;margin-bottom:0px;padding:0px;white-space:pre-wrap;font-family:'Courier New' !important;">using System; using System.Collections; namespace WebApp
{ class Program
{ static void Main(string[] args)
{
Hashtable myHash=new Hashtable(); //插入
myHash.Add("1","joye.net");
myHash.Add("2", "joye.net2");
myHash.Add("3", "joye.net3"); //key 存在
try {
myHash.Add("1", "1joye.net");
} catch {
Console.WriteLine("Key = "1" already exists.");
} //取值
Console.WriteLine("key = "2", value = {0}.", myHash["2"]); //修改
myHash["2"] = "http://www.cnblogs.com/yinrq/";
myHash["4"] = "joye.net4"; //修改的key不存在則新增
Console.WriteLine("key = "2", value = {0}.", myHash["2"]);
Console.WriteLine("key = "4", value = {0}.", myHash["4"]); //判斷key是否存在
if (!myHash.ContainsKey("5"))
{
myHash.Add("5", "joye.net5");
Console.WriteLine("key = "5": {0}", myHash["5"]);
} //移除
myHash.Remove("1"); if (!myHash.ContainsKey("1"))
{
Console.WriteLine("Key "1" is not found.");
} //foreach 取值
foreach (DictionaryEntry item in myHash)
{
Console.WriteLine("Key = {0}, Value = {1}", item.Key, item.Value);
} //所有的值
foreach (var item in myHash.Values)
{
Console.WriteLine("Value = {0}",item);
} //所有的key
foreach (var item in myHash.Keys)
{
Console.WriteLine("Key = {0}", item);
}
Console.ReadKey();
}
}
}</pre>

復(fù)制代碼

結(jié)果如下:

image

更多參考微軟官方文檔:Hashtable 類(lèi)

二、Dictionary

Dictionary<TKey, TValue> 泛型類(lèi)提供了從一組鍵到一組值的映射汁掠。通過(guò)鍵來(lái)檢索值的速度是非陈壅快的卤恳,接近于 O(1)萄喳,這是因?yàn)?Dictionary<TKey, TValue> 類(lèi)是作為一個(gè)哈希表來(lái)實(shí)現(xiàn)的。檢索速度取決于為 TKey 指定的類(lèi)型的哈希算法的質(zhì)量猾愿。TValue可以是值類(lèi)型鹦聪,數(shù)組,類(lèi)或其他蒂秘。

Dictionary是一種變種的HashTable,它采用一種分離鏈接散列表的數(shù)據(jù)結(jié)構(gòu)來(lái)解決哈希沖突的問(wèn)題泽本。

簡(jiǎn)單使用代碼:

復(fù)制代碼

<pre style="margin-top:0px;margin-bottom:0px;padding:0px;white-space:pre-wrap;font-family:'Courier New' !important;">using System; using System.Collections; using System.Collections.Generic; namespace WebApp
{ class Program
{ static void Main(string[] args)
{
Dictionary<string, string> myDic = new Dictionary<string, string>(); //插入
myDic.Add("1", "joye.net");
myDic.Add("2", "joye.net2");
myDic.Add("3", "joye.net3"); //key 存在
try {
myDic.Add("1", "1joye.net");
} catch {
Console.WriteLine("Key = "1" already exists.");
} //取值
Console.WriteLine("key = "2", value = {0}.", myDic["2"]); //修改
myDic["2"] = "http://www.cnblogs.com/yinrq/";
myDic["4"] = "joye.net4"; //修改的key不存在則新增
Console.WriteLine("key = "2", value = {0}.", myDic["2"]);
Console.WriteLine("key = "4", value = {0}.", myDic["4"]); //判斷key是否存在
if (!myDic.ContainsKey("5"))
{
myDic.Add("5", "joye.net5");
Console.WriteLine("key = "5": {0}", myDic["5"]);
} //移除
myDic.Remove("1"); if (!myDic.ContainsKey("1"))
{
Console.WriteLine("Key "1" is not found.");
} //foreach 取值
foreach (var item in myDic)
{
Console.WriteLine("Key = {0}, Value = {1}", item.Key, item.Value);
} //所有的值
foreach (var item in myDic.Values)
{
Console.WriteLine("Value = {0}",item);
} //所有的key
foreach (var item in myDic.Keys)
{
Console.WriteLine("Key = {0}", item);
}
Console.ReadKey();
}
}
}</pre>

復(fù)制代碼

運(yùn)行結(jié)果:

image

更多資料參考:Dictionary 類(lèi)

三、ConcurrentDictionary

表示可由多個(gè)線程同時(shí)訪問(wèn)的鍵/值對(duì)的線程安全集合材彪。

ConcurrentDictionary<TKey, TValue> framework4出現(xiàn)的观挎,可由多個(gè)線程同時(shí)訪問(wèn),且線程安全段化。用法同Dictionary很多相同嘁捷,但是多了一些方法。ConcurrentDictionary 屬于System.Collections.Concurrent 命名空間按照MSDN上所說(shuō):

System.Collections.Concurrent 命名空間提供多個(gè)線程安全集合類(lèi)显熏。當(dāng)有多個(gè)線程并發(fā)訪問(wèn)集合時(shí)雄嚣,應(yīng)使用這些類(lèi)代替 System.Collections 和 System.Collections.Generic 命名空間中的對(duì)應(yīng)類(lèi)型。

更多資料:ConcurrentDictionary<TKey,?TValue> 類(lèi)

四喘蟆、對(duì)比總結(jié)

分別插入500萬(wàn)條數(shù)據(jù)缓升,然后遍歷,看看耗時(shí)蕴轨。

復(fù)制代碼

<pre style="margin-top:0px;margin-bottom:0px;padding:0px;white-space:pre-wrap;font-family:'Courier New' !important;">using System; using System.Collections; using System.Collections.Concurrent; using System.Collections.Generic; using System.Diagnostics; namespace WebApp
{ class Program
{ static Hashtable _hashtable; static Dictionary<string, string> _dictionary; static ConcurrentDictionary<string, string> _conDictionary; static void Main(string[] args)
{
Compare(5000000);
Console.ReadLine();
Console.Read();
} public static void Compare(int dataCount)
{
_hashtable = new Hashtable();
_dictionary = new Dictionary<string, string>();
_conDictionary=new ConcurrentDictionary<string, string>();
Stopwatch stopWatch = new Stopwatch(); // Hashtable
stopWatch.Start(); for (int i = 0; i < dataCount; i++)
{
_hashtable.Add("key" + i.ToString(), "Value" + i.ToString());
}
stopWatch.Stop();
Console.WriteLine("HashTable插" + dataCount + "條耗時(shí)(毫秒):" + stopWatch.ElapsedMilliseconds); //Dictionary
stopWatch.Reset();
stopWatch.Start(); for (int i = 0; i < dataCount; i++)
{
_dictionary.Add("key" + i.ToString(), "Value" +i.ToString());
}
stopWatch.Stop();
Console.WriteLine("Dictionary插" + dataCount + "條耗時(shí)(毫秒):" + stopWatch.ElapsedMilliseconds); //ConcurrentDictionary
stopWatch.Reset();
stopWatch.Start(); for (int i = 0; i < dataCount; i++)
{
_conDictionary.TryAdd("key" + i.ToString(), "Value" + i.ToString());
}
stopWatch.Stop();
Console.WriteLine("ConcurrentDictionary插" + dataCount + "條耗時(shí)(毫秒):" + stopWatch.ElapsedMilliseconds); // Hashtable
stopWatch.Reset();
stopWatch.Start(); for (int i = 0; i < _hashtable.Count; i++)
{ var key = _hashtable[i];
}
stopWatch.Stop();
Console.WriteLine("HashTable遍歷時(shí)間(毫秒):" + stopWatch.ElapsedMilliseconds); //Dictionary
stopWatch.Reset();
stopWatch.Start(); for (int i = 0; i < _hashtable.Count; i++)
{ var key = _dictionary["key" + i.ToString()];
}
stopWatch.Stop();
Console.WriteLine("Dictionary遍歷時(shí)間(毫秒):" + stopWatch.ElapsedMilliseconds); //ConcurrentDictionary
stopWatch.Reset();
stopWatch.Start(); for (int i = 0; i < _hashtable.Count; i++)
{ var key = _conDictionary["key"+i.ToString()];
}
stopWatch.Stop();
Console.WriteLine("ConcurrentDictionary遍歷時(shí)間(毫秒):" + stopWatch.ElapsedMilliseconds);
}
}
}</pre>

復(fù)制代碼

運(yùn)行結(jié)果:

image

可以看出:

大數(shù)據(jù)插入Dictionary花費(fèi)時(shí)間最少

遍歷HashTable最快是Dictionary的1/5,ConcurrentDictionary的1/10

單線程建議用Dictionary港谊,多線程建議用ConcurrentDictionary或者HashTable(Hashtable tab = Hashtable.Synchronized(new Hashtable());獲得線程安全的對(duì)象)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市橙弱,隨后出現(xiàn)的幾起案子歧寺,更是在濱河造成了極大的恐慌,老刑警劉巖棘脐,帶你破解...
    沈念sama閱讀 217,509評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件斜筐,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡蛀缝,警方通過(guò)查閱死者的電腦和手機(jī)顷链,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)屈梁,“玉大人嗤练,你說(shuō)我怎么就攤上這事榛了。” “怎么了潭苞?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,875評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵忽冻,是天一觀的道長(zhǎng)真朗。 經(jīng)常有香客問(wèn)我此疹,道長(zhǎng),這世上最難降的妖魔是什么遮婶? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,441評(píng)論 1 293
  • 正文 為了忘掉前任蝗碎,我火速辦了婚禮,結(jié)果婚禮上旗扑,老公的妹妹穿的比我還像新娘蹦骑。我一直安慰自己,他們只是感情好臀防,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,488評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布眠菇。 她就那樣靜靜地躺著,像睡著了一般袱衷。 火紅的嫁衣襯著肌膚如雪捎废。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,365評(píng)論 1 302
  • 那天致燥,我揣著相機(jī)與錄音登疗,去河邊找鬼。 笑死嫌蚤,一個(gè)胖子當(dāng)著我的面吹牛辐益,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播脱吱,決...
    沈念sama閱讀 40,190評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼智政,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了箱蝠?” 一聲冷哼從身側(cè)響起续捂,我...
    開(kāi)封第一講書(shū)人閱讀 39,062評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎抡锈,沒(méi)想到半個(gè)月后疾忍,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,500評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡床三,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,706評(píng)論 3 335
  • 正文 我和宋清朗相戀三年一罩,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片撇簿。...
    茶點(diǎn)故事閱讀 39,834評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡聂渊,死狀恐怖差购,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情汉嗽,我是刑警寧澤欲逃,帶...
    沈念sama閱讀 35,559評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站饼暑,受9級(jí)特大地震影響稳析,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜弓叛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,167評(píng)論 3 328
  • 文/蒙蒙 一彰居、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧撰筷,春花似錦陈惰、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,779評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至关筒,卻和暖如春溶握,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背平委。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,912評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工奈虾, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人廉赔。 一個(gè)月前我還...
    沈念sama閱讀 47,958評(píng)論 2 370
  • 正文 我出身青樓肉微,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親蜡塌。 傳聞我的和親對(duì)象是個(gè)殘疾皇子碉纳,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,779評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • ArrayList (非泛型集合 using System.Collections;) public void T...
    Unity開(kāi)發(fā)閱讀 349評(píng)論 1 0
  • 一劳曹、什么是Hashtable? Hashtable 類(lèi)代表了一系列基于鍵的哈希代碼組織起來(lái)的鍵/值對(duì)琅摩。它使用鍵來(lái)訪...
    小明yz閱讀 2,615評(píng)論 0 3
  • HashTable的應(yīng)用非常廣泛铁孵,HashMap是新框架中用來(lái)代替HashTable的類(lèi),也就是說(shuō)建議使用Hash...
    RoboyCore閱讀 512評(píng)論 0 2
  • 小珺: 展信好房资! 這幾天幾乎每天晚上都有給你寫(xiě)信蜕劝,已經(jīng)成為我晚上最幸福的事情了。昨天晚上開(kāi)會(huì)到了十二點(diǎn),寫(xiě)完第三十...
    風(fēng)不眠閱讀 128評(píng)論 0 0
  • 你光芒四射 讓人不敢直視 卻照亮著前方 日出東方 帶來(lái)美好的開(kāi)始和希翼 日落西山 柔美的霞光無(wú)盡的想像 陰郁的曰子...
    洛水秦韻閱讀 456評(píng)論 0 19