HashTable和HashMap的区别

    技术2022-05-19  21

    HashTable的应用非常广泛,HashMap是新框架中用来代替HashTable的类,也就是说建议使用HashMap,不要使用HashTable。可能你觉得HashTable很好用,为什么不用呢?这里简单分析他们的区别。

    1.HashTable的方法是同步的,HashMap未经同步,所以在多线程场合要手动同步HashMap这个区别就像Vector和ArrayList一样。

    2.HashTable不允许null值(key和value都不可以),HashMap允许null值(key和value都可以)。

    3.HashTable有一个contains(Object value),功能和containsValue(Object value)功能一样。

    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值,而且用与代替求模:int hash = hash(k);int i = indexFor(hash, table.length);static int hash(Object x) {  int h = x.hashCode();

      h += ~(h << 9);  h ^= (h >>> 14);  h += (h << 4);  h ^= (h >>> 10);  return h;}static int indexFor(int h, int length) {  return h & (length-1);}以上只是一些比较突出的区别,当然他们的实现上还是有很多不同的,比如HashMap对null的操作。

    补充:

     

    HashMap中,null可以作为键,这样的键只有一个;可以有一个或多个键所对应的值为null。当get()方法返回null值时,即可以表示 HashMap中没有该键,也可以表示该键所对应的值为null。因此,在HashMap中不能由get()方法来判断HashMap中是否存在某个键, 而应该用containsKey()方法来判断。

    不用多说,看下面的程序就可以:

    HashMap map = new HashMap();

    map.put("Null", null);

    map.put(null, "Null");

    map.put(null, "Empty");

    System.out.println(map.get(null));

    System.out.println(map.get("Null"));

    System.out.println(map.get("NullThere"));

    System.out.println(map.containsKey("Null"));

    System.out.println(map.containsKey("NullThere"));

     

     

    输出结果为:

    Empty

    null

    null

    true

    false

    HashMap

    Hashtable

    继承,实现

    HashMap<K,V>    extends AbstractMap<K,V>    implements Map<K,V>, Cloneable, Serializable

    Hashtable<K,V>

        extends Dictionary<K,V>

        implements Map<K,V>, Cloneable,Serializable

    多线程,同步

    未同步的,可以使用Colletcions进行同步

    Map Collections.synchronizedMap(Map m)

    已经同步过的可以安全使用

    null的处理

    HashMap map = new HashMap();

    map.put(null, "Null");

    map.put("Null", null);

    map.containsKey(null);

    map.containsValue(null);

    以上这5条语句无论在编译期,还是在运行期都是没有错误的.

    HashMap中,null可以作为键,这样的键只有一个;可以有一个或多个键所对应的值为null。当get()方法返回null值时,即可以表示 HashMap中没有该键,也可以表示该键所对应的值为null。因此,在HashMap中不能由get()方法来判断HashMap中是否存在某个键, 而应该用containsKey()方法来判断。

    Hashtable table = new Hashtable();

    table.put(null, "Null");

    table.put("Null", null);

    table.contains(null);

    table.containsKey(null);

    table.containsValue(null);

    后面的5句话在编译的时候不会有异常,可在运行的时候会报空指针异常

    具体原因可以查看源代码

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

         // Make sure the value is not null

         if (value == null) {

             throw new NullPointerException();

         }

    ………….

    增长率

    void addEntry(int hash, K key, V value, int bucketIndex) {

          Entry<K,V> e = table[bucketIndex];

            table[bucketIndex] = new Entry<K,V>(hash, key, value, e);

            if (size++ >= threshold)

                resize(2 * table.length);

        }

    protected void rehash() {

          int oldCapacity = table.length;

          Entry[] oldMap = table;

          int newCapacity = oldCapacity * 2 + 1;

          Entry[] newMap = new Entry[newCapacity];

          modCount++;

          threshold = (int)(newCapacity * loadFactor);

          table = newMap;

          for (int i = oldCapacity ; i-- > 0 ;) {

              for (Entry<K,V> old = oldMap[i] ; old != null ; ) {

               Entry<K,V> e = old;

               old = old.next;

               int index = (e.hash & 0x7FFFFFFF) % newCapacity;

               e.next = newMap[index];

               newMap[index] = e;

              }

          }

        }

     

    哈希值的使用

    HashMap重新计算hash值,而且用与代替求模

          public boolean containsKey(Object key) {

                  Object k = maskNull(key);

                  int hash = hash(k.hashCode());

                  int i = indexFor(hash, table.length);

                  Entry e = table[i];

                  while (e != null) {

                      if (e.hash == hash && eq(k, e.key))

                          return true;

                      e = e.next;

                  }

                  return false;

              }

    HashTable直接使用对象的hashCode,代码是这样的:

    public synchronized boolean containsKey(Object key) {

          Entry tab[] = table;

          int hash = key.hashCode();

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

          for (Entry<K,V> e = tab[index] ; e != null ; e = e.next) {

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

               return true;

              }

          }

          return false;

        }

     


    最新回复(0)