HashMap和HashTable的區(qū)別

HashMap和Hashtable的比較是Java面試中的常見問題,用來考驗程序員是否能夠正確使用集合類以及是否可以隨機應變使用多種思路解決問題。HashMap的工作原理腹缩、ArrayList與Vector的比較以及這個問題是有關Java 集合框架的最經(jīng)典的問題。Hashtable是個過時的集合類,存在于Java API中很久了如绸。在Java 4中被重寫了,實現(xiàn)了Map接口旭贬,所以自此以后也成了Java集合框架中的一部分怔接。Hashtable和HashMap在Java面試中相當容易被問到,甚至成為了集合框架面試題中最常被考的問題稀轨,所以在參加任何Java面試之前扼脐,都不要忘了準備這一題。

HashMap和Hashtable的區(qū)別

HashMap和Hashtable都實現(xiàn)了Map接口靶端,但決定用哪一個之前先要弄清楚它們之間的分別谎势。主要的區(qū)別有:線程安全性凛膏,同步(synchronization),以及速度脏榆。

一猖毫、繼承的接口和類不同

public class HashMap extends AbstractMap implements Map, Cloneable, Serializable{} 

public class Hashtable extends Dictionary implements Map, Cloneable, java.io.Serializable{} 

二、線程安全

HashMap幾乎可以等價于Hashtable须喂,除了HashMap是非synchronized的吁断,HashTable中的方法是同步的。故Hashtable是線程安全的坞生,多個線程可以共享一個Hashtable仔役;而如果沒有正確的同步的話,多個線程是不能共享HashMap的是己。Java 5提供了ConcurrentHashMap又兵,它是HashTable的替代,比HashTable的擴展性更好卒废。

HashTable中的put()方法:

public synchronized V put(K key,V value) {

// Make sure the value is not null

    if(value==null) {

        throw newNullPointerException();

    }

    // Makes sure the key is not already in the hashtable.

    Entry tab[] =table;

    int hash=key.hashCode();

    int index= (hash&0x7FFFFFFF) %tab.length;

    @SuppressWarnings("unchecked")

    Entry entry= (Entry)tab[index];

    for(;entry!=null;entry=entry.next) {

        if((entry.hash==hash) &&entry.key.equals(key)) {

        V old=entry.value;

        entry.value=value;

        return old;

        }

    }

    addEntry(hash,key,value,index);

    return null;

}

HashMap中的put()方法:

public V put(K key,V value) {

    return putVal(hash(key), key, value, false, true);

}

三沛厨、null的處理:

HashMap 可以接受null(HashMap可以接受為null的鍵值(key)和值(value),而Hashtable則不行)摔认,其中只有一個key可以為null逆皮,多個不同的key對應的value可以為null。判斷是否存在某個key應該用containsKey(key)参袱。

HashTable key和value都不能為null电谣,否則拋出異常NullPointerException

四、遍歷方式略有差異

HahsMap,Hashtable都可使用Iterator遍歷:

Collectioncollection=map.values();

Iteratorit=collection.iterator();

CollectioncollectionTable=table.values();

IteratoritTable=collectionTable.iterator();

Hashtable還使用了Enumeration的方式:

public synchronized Enumeration elements() {

    return this.getEnumeration(VALUES);

}

兩者也都可以通過 entrySet() 方法返回一個 Set 抹蚀, 然后進行遍歷處理:

HashMap:

Set setMap=map.entrySet();

Iterator itMapSet=setMap.iterator();

while(itMapSet.hasNext()) {

    Map.Entryentry= (Map.Entry)itMapSet.next();

    System.out.println("entry.getKey()  = "+entry.getKey() +", entry.getValue() = "+entry.getValue() );

}

HashTable:

    Set setTable=table.entrySet();

HashMap的迭代器(Iterator)是fail-fast迭代器剿牺,而Hashtable的enumerator迭代器不是fail-fast的。所以當有其它線程改變了HashMap的結構(增加或者移除元素)环壤,將會拋出ConcurrentModificationException牢贸,但迭代器本身的remove()方法移除元素則不會拋出ConcurrentModificationException異常。但這并不是一個一定發(fā)生的行為镐捧,要看JVM潜索。這條同樣也是Enumeration和Iterator的區(qū)別。

五懂酱、哈希值的使用不同竹习,Hashtable直接使用對象的hashCode:

int hash=key.hashCode();

而HashMap重新計算hash值:

static final int hash(Objectkey) {

    int h;

    return(key==null) ?0: (h=key.hashCode()) ^ (h>>>16);

}

六、Hashtable中hash數(shù)組默認大小是11列牺,增加的方式是 old*2+1整陌。HashMap中hash數(shù)組的默認大小是16,而且一定是2的指數(shù)。

HashMap:

/**

* The default initial capacity - MUST be a power of two.

*/

static final int DEFAULT_INITIAL_CAPACITY=1<<4;// aka 16

HashTable:

// overflow-conscious code

intnewCapacity= (oldCapacity<<1) +1;

要注意的一些重要術語:

  1. sychronized意味著在一次僅有一個線程能夠更改Hashtable泌辫。就是說任何線程要更新Hashtable時要首先獲得同步鎖随夸,其它線程要等到同步鎖被釋放之后才能再次獲得同步鎖更新Hashtable。

  2. Fail-safe和iterator迭代器相關震放。如果某個集合對象創(chuàng)建了Iterator或者ListIterator宾毒,然后其它的線程試圖“結構上”更改集合對象,將會拋出ConcurrentModificationException異常殿遂。但其它線程可以通過set()方法更改集合對象是允許的诈铛,因為這并沒有從“結構上”更改集合。但是假如已經(jīng)從結構上進行了更改墨礁,再調(diào)用set()方法履怯,將會拋出IllegalArgumentException異常绷杜。

  3. 結構上的更改指的是刪除或者插入一個元素,這樣會影響到map的結構搂誉。

我們能否讓HashMap同步癣丧?

HashMap可以通過下面的語句進行同步:

Map m = Collections.synchronizedMap(hashMap);

結論

Hashtable和HashMap有幾個主要的不同:線程安全以及速度绿聘。僅在你需要完全的線程安全的時候使用Hashtable劲厌,而如果你使用Java 5或以上的話路呜,請使用ConcurrentHashMap吧。

參考:http://www.importnew.com/7010.html

http://www.importnew.com/25070.html

http://www.importnew.com/24822.html

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末轻掩,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子懦底,更是在濱河造成了極大的恐慌唇牧,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件聚唐,死亡現(xiàn)場離奇詭異丐重,居然都是意外死亡,警方通過查閱死者的電腦和手機杆查,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門扮惦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人亲桦,你說我怎么就攤上這事崖蜜。” “怎么了客峭?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵豫领,是天一觀的道長。 經(jīng)常有香客問我舔琅,道長等恐,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮课蔬,結果婚禮上囱稽,老公的妹妹穿的比我還像新娘。我一直安慰自己二跋,他們只是感情好战惊,可當我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著同欠,像睡著了一般样傍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上铺遂,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天衫哥,我揣著相機與錄音,去河邊找鬼襟锐。 笑死撤逢,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的粮坞。 我是一名探鬼主播蚊荣,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼莫杈!你這毒婦竟也來了互例?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤筝闹,失蹤者是張志新(化名)和其女友劉穎媳叨,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體关顷,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡糊秆,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了议双。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片痘番。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖平痰,靈堂內(nèi)的尸體忽然破棺而出汞舱,到底是詐尸還是另有隱情,我是刑警寧澤宗雇,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布兵拢,位于F島的核電站,受9級特大地震影響逾礁,放射性物質(zhì)發(fā)生泄漏说铃。R本人自食惡果不足惜访惜,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望腻扇。 院中可真熱鬧债热,春花似錦、人聲如沸幼苛。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽舶沿。三九已至墙杯,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間括荡,已是汗流浹背高镐。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留畸冲,地道東北人嫉髓。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓,卻偏偏與公主長得像邑闲,于是被迫代替她去往敵國和親算行。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,979評論 2 355

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