1、Hashtable是Dictionary的子類,
複製代碼代碼如下:
public class Hashtable<K,V>
extends Dictionary<K,V>
implements Map<K,V>, Cloneable, java.io.Serializable
HashMap:
複製代碼代碼如下:
public class HashMap<K,V>
extends AbstractMap<K,V>
implements Map<K,V>, Cloneable, Serializable
HashMap和Hashtable都是Map介面的實作類別;
2、Hashtable中的方法是同步的(),而HashMap中的方法在預設情況下不是同步的。即是說,在多執行緒應用程式中,不用專門的操作就安全地可以使用Hashtable了;而對於HashMap,則需要額外的同步機制。但HashMap的同步問題可透過Collections的一個靜態方法來解決:
複製代碼代碼如下:
public static <K,V> Map<K,V> synchronizedMap(Map<K,V> m)
這個方法回傳一個同步的Map,也就是說傳回的Map是線程安全的。需要注意的是,對傳回的map進行迭代時,必須手動在傳回的map上進行同步,否則將會導致不確定的行為:
複製代碼代碼如下:
Map m = Collections.synchronizedMap(new HashMap());
…
Set s = m.keySet(); // Needn't be in synchronized block
…
synchronized(m) { // Synchronizing on m, not s!
Iterator i = s.iterator(); // Must be in synchronized block
while (i.hasNext())
foo(i.next());
}
3.在HashMap中,null可以作為鍵,這樣的鍵只有一個;可以有一個或多個鍵所對應的值為null。當get()方法傳回null值時,也就是可以表示HashMap中沒有該鍵,也可以表示該鍵所對應的值為null。因此,在HashMap中不能由get()方法來判斷HashMap中是否存在某個鍵,而應該用containsKey()方法來判斷。 Hashtable的鍵值不能為null,否則:java.lang.NullPointerException 。
4.HashTable使用Enumeration,HashMap使用Iterator。
以上只是表面的不同,它們的實現也有很大的不同。
5.HashTable中hash數組預設大小是11,增加的方式是old*2+1。 HashMap中hash數組的預設大小是16,而且一定是2的指數。
6.哈希值的使用不同,HashTable直接使用物件的hashCode,程式碼是這樣的:
複製代碼代碼如下:
int hash = key.hashCode();
int index = (hash & 0x7FFFFFFF) % tab.length;
而HashMap重新計算hash值,用與取代求模,例如HashMap的put方法:
複製代碼代碼如下:
public V put(K key, V value) {
if (key == null)
return putForNullKey(value);
int hash = hash(key.hashCode());
int i = indexFor(hash, table.length);
for (Entry<K,V> e = table[i]; e != null; e = e.next) {
Object k;
if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
V oldValue = e.value;
e.value = value;
e.recordAccess(this);
return oldValue;
}
}
modCount++;
addEntry(hash, key, value, i);
return null;
}
複製代碼代碼如下:
static int hash(int h) {
// This function ensures that hashCodes that differ only by
// constant multiples at each bit position have a bounded
// number of collisions (approximately 8 at default load factor).
h ^= (h >>> 20) ^ (h >>> 12);
return h ^ (h >>> 7) ^ (h >>> 4);
}
複製代碼代碼如下:
static int indexFor(int h, int length) {
return h & (length-1);
}