Hashtable是個(gè)過(guò)時(shí)的集合類匆笤,存在于Java API中很久了谱邪。在Java 4中被重寫(xiě)了,實(shí)現(xiàn)了Map接口惦银,所以自此以后也成了Java集合框架中的一部分。
HashMap和Hashtable都實(shí)現(xiàn)了Map接口书蚪,但決定用哪一個(gè)之前先要弄清楚它們之間的分別殊校。主要的區(qū)別有:線程安全性,同步(synchronization)为流,以及速度。
HashMap和Hashtable的區(qū)別
- HashMap幾乎可以等價(jià)于Hashtable艺谆,除了HashMap是非synchronized的静汤,并可以接受null(HashMap可以接受為null的鍵值(key)和值(value),而Hashtable則不行)藤抡。
- HashMap是非synchronized,而Hashtable是synchronized缠黍,這意味著Hashtable是線程安全的药蜻,多個(gè)線程可以共享一個(gè)Hashtable语泽;而如果沒(méi)有正確的同步的話,多個(gè)線程是不能共享HashMap的踱卵。Java 5提供了ConcurrentHashMap,它是HashTable的替代妒挎,比HashTable的擴(kuò)展性更好酝掩。
- 另一個(gè)區(qū)別是HashMap的迭代器(Iterator)是fail-fast迭代器罗标,而Hashtable的enumerator迭代器不是fail-fast的。所以當(dāng)有其它線程改變了HashMap的結(jié)構(gòu)(增加或者移除元素)彻消,將會(huì)拋出ConcurrentModificationException宙拉,但迭代器本身的remove()方法移除元素則不會(huì)拋出ConcurrentModificationException異常。但這并不是一個(gè)一定發(fā)生的行為煌贴,要看JVM。這條同樣也是Enumeration和Iterator的區(qū)別牛郑。
- 由于Hashtable是線程安全的也是synchronized淹朋,所以在單線程環(huán)境下它比HashMap要慢。如果你不需要同步杈抢,只需要單一線程,那么使用HashMap性能要好過(guò)Hashtable惶楼。
- HashMap不能保證隨著時(shí)間的推移Map中的元素次序是不變的诊杆。
要注意的一些重要術(shù)語(yǔ):
sychronized意味著在一次僅有一個(gè)線程能夠更改Hashtable。就是說(shuō)任何線程要更新Hashtable時(shí)要首先獲得同步鎖窥岩,其它線程要等到同步鎖被釋放之后才能再次獲得同步鎖更新Hashtable宰缤。
Fail-safe和iterator迭代器相關(guān)慨灭。如果某個(gè)集合對(duì)象創(chuàng)建了Iterator或者ListIterator氧骤,然后其它的線程試圖“結(jié)構(gòu)上”更改集合對(duì)象,將會(huì)拋出ConcurrentModificationException異常筹陵。但其它線程可以通過(guò)set()方法更改集合對(duì)象是允許的朦佩,因?yàn)檫@并沒(méi)有從“結(jié)構(gòu)上”更改集合庐氮。但是假如已經(jīng)從結(jié)構(gòu)上進(jìn)行了更改,再調(diào)用set()方法,將會(huì)拋出IllegalArgumentException異常输涕。
結(jié)構(gòu)上的更改指的是刪除或者插入一個(gè)元素莱坎,這樣會(huì)影響到map的結(jié)構(gòu)檐什。
我們能否讓HashMap同步?
HashMap可以通過(guò)下面的語(yǔ)句進(jìn)行同步:
Map m = Collections.synchronizeMap(hashMap);
結(jié)論
Hashtable和HashMap有幾個(gè)主要的不同:線程安全以及速度厢汹。僅在你需要完全的線程安全的時(shí)候使用Hashtable烫葬,而如果你使用Java 5或以上的話凡蜻,請(qǐng)使用ConcurrentHashMap吧划栓。