HashMap和Hashtable的區(qū)別
HashMap和Hashtable都實現(xiàn)了Map接口扎即,但決定用哪一個之前先要弄清楚它們之間的分別。主要的區(qū)別有:線程安全性况凉,同步(synchronization)谚鄙,以及速度刁绒。
HashMap幾乎可以等價于Hashtable闷营,除了HashMap是非synchronized的,并可以接受null(HashMap可以接受為null的鍵值(key)和值(value)知市,而Hashtable則不行)。
HashMap是非synchronized嫂丙,而Hashtable是synchronized娘赴,這意味著Hashtable是線程安全的,多個線程可以共享一個Hashtable跟啤;而如果沒有正確的同步的話诽表,多個線程是不能共享HashMap的。Java 5提供了ConcurrentHashMap腥光,它是HashTable的替代关顷,比HashTable的擴展性更好武福。
另一個區(qū)別是HashMap的迭代器(Iterator)是fail-fast迭代器议双,而Hashtable的enumerator迭代器不是fail-fast的。所以當(dāng)有其它線程改變了HashMap的結(jié)構(gòu)(增加或者移除元素)捉片,將會拋出ConcurrentModificationException,但迭代器本身的remove()方法移除元素則不會拋出ConcurrentModificationException異常伍纫。但這并不是一個一定發(fā)生的行為宗雇,要看JVM。這條同樣也是Enumeration和Iterator的區(qū)別莹规。
由于Hashtable是線程安全的也是synchronized赔蒲,所以在單線程環(huán)境下它比HashMap要慢。如果你不需要同步,只需要單一線程舞虱,那么使用HashMap性能要好過Hashtable。
HashMap不能保證隨著時間的推移Map中的元素次序是不變的矾兜。
要注意的一些重要術(shù)語:
sychronized意味著在一次僅有一個線程能夠更改Hashtable损趋。就是說任何線程要更新Hashtable時要首先獲得同步鎖,其它線程要等到同步鎖被釋放之后才能再次獲得同步鎖更新Hashtable椅寺。
Fail-safe和iterator迭代器相關(guān)浑槽。如果某個集合對象創(chuàng)建了Iterator或者ListIterator,然后其它的線程試圖“結(jié)構(gòu)上”更改集合對象返帕,將會拋出ConcurrentModificationException異常。但其它線程可以通過set()方法更改集合對象是允許的荆萤,因為這并沒有從“結(jié)構(gòu)上”更改集合畸冲。但是假如已經(jīng)從結(jié)構(gòu)上進行了更改,再調(diào)用set()方法观腊,將會拋出IllegalArgumentException異常。
結(jié)構(gòu)上的更改指的是刪除或者插入一個元素算行,這樣會影響到map的結(jié)構(gòu)梧油。
我們能否讓HashMap同步?
HashMap可以通過下面的語句進行同步:
Map m = Collections.synchronizeMap(hashMap);
結(jié)論
Hashtable和HashMap有幾個主要的不同:線程安全以及速度儡陨。僅在你需要完全的線程安全的時候使用Hashtable,而如果你使用Java 5或以上的話量淌,請使用ConcurrentHashMap吧骗村。