Java8集合源码解析-Hashtable源码剖析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 欢迎志同道合的小伙伴一起交流Java学习,共同应对校招点击链接加入群【java编程技术交流】:https://jq.qq.com/?_wv=1027&k=4A14H0S1 概述本文将介绍Map集合的另一个常用类,Hashtable.

欢迎志同道合的小伙伴一起交流Java学习,共同应对校招
点击链接加入群【java编程技术交流】:https://jq.qq.com/?_wv=1027&k=4A14H0S

1 概述

本文将介绍Map集合的另一个常用类,Hashtable.Hashtable出来的比HashMap早,HashMap1.2才有,而Hashtable在1.0就已经出现了.HashMap和Hashtable实现原理基本一样,都是通过哈希表实现.而且两者处理冲突的方式也一样,都是通过链表法.下面就详细学习下这个类.

2 源码解析

类总览

public class Hashtable<K,V>
    extends Dictionary<K,V>
    implements Map<K,V>, Cloneable, java.io.Serializable

Hashtable并没有去继承AbstractMap,而是选择继承了Dictionary类,Dictionary是个被废弃的抽象类,文档已经说得很清楚了:

NOTE: This class is obsolete.  New implementations should  
 * implement the Map interface, rather than extending this class.  

这个类的方法如下(全是抽象方法):

public abstract  
class Dictionary<K,V> {  
    
    public Dictionary() {  
    }  
    abstract public int size();  
    abstract public boolean isEmpty();  
    abstract public Enumeration<K> keys();  
    abstract public Enumeration<V> elements();  
    abstract public V get(Object key);  
    abstract public V put(K key, V value);  
    abstract public V remove(Object key);  
}  

成员变量

   //存储键值对的桶数组  
   private transient Entry<K,V>[] table;

   //键值对总数 
   private transient int count;  
  
   //容量的阈值,超过此容量将会导致扩容。值为容量*负载因子    
   private int threshold;  
   
   //负载因子    
   private float loadFactor;  
   
   //hashtable被改变的次数,用于快速失败机制 
   private transient int modCount = 0;  

成员变量跟HashMap基本类似,但是HashMap更加规范,HashMap内部还定义了一些常量,比如默认的负载因子,默认的容量,最大容量等等。

构造方法

//可指定初始容量和加载因子  
public Hashtable(int initialCapacity, float loadFactor) {
        if (initialCapacity < 0)  
            throw new IllegalArgumentException("Illegal Capacity: "+  
                                               initialCapacity);  
        if (loadFactor <= 0 || Float.isNaN(loadFactor))  
            throw new IllegalArgumentException("Illegal Load: "+loadFactor);  
        if (initialCapacity==0) 
            //初始容量最小值为1  
            initialCapacity = 1; 
        this.loadFactor = loadFactor;  
        //创建桶数组
        table = new Entry[initialCapacity];  
        //初始化容量阈值  
        threshold = (int)Math.min(initialCapacity * loadFactor, MAX_ARRAY_SIZE + 1);
        useAltHashing = sun.misc.VM.isBooted() &&  
                (initialCapacity >= Holder.ALTERNATIVE_HASHING_THRESHOLD);  
    }  
    
    public Hashtable(int initialCapacity) { 
        //默认负载因子为0.75   
        this(initialCapacity, 0.75f);
    }  
    public Hashtable() {  
        this(11, 0.75f);//默认容量为11,负载因子为0.75  
    }  
   
    public Hashtable(Map<? extends K, ? extends V> t) {  
        this(Math.max(2*t.size(), 11), 0.75f);  
        putAll(t);  
    }  
  1. Hashtable的默认容量为11,默认负载因子为0.75.(HashMap默认容量为16,默认负载因子也是0.75)
  2. Hashtable的容量可以为任意整数,最小值为1,而HashMap的容量始终为2的n次方
  3. 为避免扩容带来的性能问题,建议指定合理容量。

另外我们看到,Hashtable的编码相比较HashMap不是很规范,构造器中出现了硬编码,而HashMap中定义了常量。

跟HashMap一样,Hashtable内部也有一个静态类叫Entry,其实是个键值对对象,保存了键和值的引用。也可以理解为一个单链表的结点,因为其持有下一个Entry对象的引用:

Entry类

private static class Entry<K,V> implements Map.Entry<K,V> {//键值对对象  
        int hash;  
        final K key;  
        V value;  
        Entry<K,V> next;  
        protected Entry(int hash, K key, V value, Entry<K,V> next) {  
            this.hash = hash;  
            this.key =  key;  
            this.value = value;  
            this.next = next;  
        }  
        protected Object clone() {  
            return new Entry<>(hash, key, value,  
                                  (next==null ? null : (Entry<K,V>) next.clone()));  
        }  
        // Map.Entry Ops  
        public K getKey() {  
            return key;  
        }  
        public V getValue() {  
            return value;  
        }  
        public V setValue(V value) {  
            if (value == null)  
                throw new NullPointerException();  
            V oldValue = this.value;  
            this.value = value;  
            return oldValue;  
        }  
        public boolean equals(Object o) {  
            if (!(o instanceof Map.Entry))  
                return false;  
            Map.Entry<?,?> e = (Map.Entry)o;  
            return key.equals(e.getKey()) && value.equals(e.getValue());  
        }  
        public int hashCode() {  
            return hash ^ value.hashCode();  
        }  
        public String toString() {  
            return key.toString()+"="+value.toString();  
        }  
    }  

再次强调:HashMap和Hashtable存储的是键值对对象,而不是单独的键或值。
明确了存储方式后,再看put和get方法:

put方法

//向哈希表中添加键值对
public synchronized V put(K key, V value) {  
        //确保值不为空 
        if (value == null) {  
            throw new NullPointerException();  
        }  
        // Makes sure the key is not already in the hashtable.  
        Entry tab[] = table; 
        //生成键的hash值,若key为null,此方法会抛异常   
        int hash = hash(key);
        //通过hash值找到其存储位置  
        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)) {
                V old = e.value;  
                e.value = value;  
                return old;  
            }  
        }  
        modCount++;  
        //当前容量超过阈值,需要扩容 
        if (count >= threshold) { 
            //重新构建桶数组,并对数组中所有键值对重哈希,耗时!        
            rehash();  
            tab = table;  
            hash = hash(key);  
            index = (hash & 0x7FFFFFFF) % tab.length;  
        }   
        Entry<K,V> e = tab[index];  
        //将新结点插到链表首部  
        tab[index] = new Entry<>(hash, key, value, e);//生成一个新结点  
        count++;  
        return null;  
    }

Hasbtable并不允许值和键为空(null),若为空,会抛空指针.大家可能奇怪,put方法在开始处仅对value进行判断,并未对key判断,可能是设计者的疏忽。当然,这并不影响使用,因为当调用hash方法时,若key为空,依然会抛出空指针异常:

private int hash(Object k) {  
        if (useAltHashing) {  
            if (k.getClass() == String.class) {  
                return sun.misc.Hashing.stringHash32((String) k);  
            } else {  
                int h = hashSeed ^ k.hashCode();  
                h ^= (h >>> 20) ^ (h >>> 12);  
                return h ^ (h >>> 7) ^ (h >>> 4);  
             }  
        } else  {  
            return k.hashCode();//此处可能抛空指针异常  
        }  
    }  
  • HashMap计算索引的方式是h&(length-1),而Hashtable用的是模运算,效率上是低于HashMap的
  • Hashtable计算索引时将hash值先与上0x7FFFFFFF,这是为了保证hash值始终为正数
  • 特别需要注意的是这个方法包括下面要讲的若干方法都加了synchronized关键字,也就意味着这个Hashtable是个线程安全的类,这也是它和HashMap最大的不同点

rehash方法

protected void rehash() { 
        //记录旧容量  
        int oldCapacity = table.length; 
        //记录旧的桶数组   
        Entry<K,V>[] oldMap = table;
        //可能会溢出 
        //扩容为原容量的2倍加1
        int newCapacity = (oldCapacity << 1) + 1;  
        if (newCapacity - MAX_ARRAY_SIZE > 0) {  
            if (oldCapacity == MAX_ARRAY_SIZE)
                return; 
            //容量不得超过约定的最大值,若超过依旧保持为约定的最大值     
            newCapacity = MAX_ARRAY_SIZE;  
        }  
        //创建新的数组  
        Entry<K,V>[] newMap = new Entry[newCapacity];
        //结构改变计数器加一
        modCount++;  
        threshold = (int)Math.min(newCapacity * loadFactor, MAX_ARRAY_SIZE + 1);  
        boolean currentAltHashing = useAltHashing;  
        useAltHashing = sun.misc.VM.isBooted() &&  
                (newCapacity >= Holder.ALTERNATIVE_HASHING_THRESHOLD);  
        boolean rehash = currentAltHashing ^ useAltHashing;  
        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;  
                if (rehash) {  
                    e.hash = hash(e.key);  
                }  
                int index = (e.hash & 0x7FFFFFFF) % newCapacity;  
                e.next = newMap[index];  
                newMap[index] = e;  
            }  
        }  
    } 

Hashtable每次扩容,容量都为原来的2倍加1,而HashMap为原来的2倍.

get方法

public synchronized V get(Object key) {//根据键取出对应索引  
      Entry tab[] = table;  
      int hash = hash(key);//先根据key计算hash值  
      int index = (hash & 0x7FFFFFFF) % tab.length;//再根据hash值找到索引  
      for (Entry<K,V> e = tab[index] ; e != null ; e = e.next) {//遍历entry链  
          if ((e.hash == hash) && e.key.equals(key)) {//若找到该键  
              return e.value;//返回对应的值  
          }  
      }  
      return null;//否则返回null  
  } 

remove方法

public synchronized V remove(Object key) {//删除指定键值对  
       Entry tab[] = table;  
       int hash = hash(key);//计算hash值  
       int index = (hash & 0x7FFFFFFF) % tab.length;//计算索引  
       for (Entry<K,V> e = tab[index], prev = null ; e != null ; prev = e, e = e.next) {//遍历entry链  
           if ((e.hash == hash) && e.key.equals(key)) {//找到指定键  
               modCount++;  
               if (prev != null) {//修改相关指针  
                   prev.next = e.next;  
               } else {  
                   tab[index] = e.next;  
               }  
               count--;  
               V oldValue = e.value;  
               e.value = null;  
               return oldValue;  
           }  
       }  
       return null;  
   } 

clear方法

//清空桶数组  
public synchronized void clear() {
       Entry tab[] = table;  
       modCount++;  
       for (int index = tab.length; --index >= 0; )  
           tab[index] = null;//直接置空  
       count = 0;  
   }  

获取键集(keySet)和键值集(entrySet)的方法

public Set<K> keySet() {  
        if (keySet == null)//通过Collections的包装,返回的是线程安全的键集  
            keySet = Collections.synchronizedSet(new KeySet(), this);  
        return keySet;  
    }  
 public Set<Map.Entry<K,V>> entrySet() {  
        if (entrySet==null)//通过Collections的包装,返回的是线程安全的键值集  
            entrySet = Collections.synchronizedSet(new EntrySet(), this);  
        return entrySet;  
    }  

这个KeySet和EntrySet是Hashtable的两个内部类:

private class KeySet extends AbstractSet<K> {  
       public Iterator<K> iterator() {  
           return getIterator(KEYS);  
       }  
       public int size() {  
           return count;  
       }  
       public boolean contains(Object o) {  
           return containsKey(o);  
       }  
       public boolean remove(Object o) {  
           return Hashtable.this.remove(o) != null;  
       }  
       public void clear() {  
           Hashtable.this.clear();  
       }  
   }  

3 总结

  1. Hashtable是线程安全的类(HashMap非线程安全)
  2. HashTable不允许null值(key和value都不可以) ;HashMap允许null值(key和value都可以)
  3. HashTable有一个contains(Object value)功能和containsValue(Object
    value)功能一样
  4. Hashtable不允许键重复,若键重复,则新插入的值会覆盖旧值(同HashMap)
  5. HashTable使用Enumeration进行遍历;HashMap使用Iterator进行遍历
  6. Hashtable同样是通过链表法解决冲突
  7. 哈希值的使用不同,HashTable直接使用对象的hashCode; HashMap重新计算hash值,而且用与代替求模
  8. Hashtable根据hashCode()计算索引时将hash值先与上0x7FFFFFFF,这是为了保证hash值始终为正数
  9. Hashtable的容量为任意正数(最小为1),而HashMap的容量始终为2的n次方.Hashtable默认容量为 11,HashMap默认容量为16
  10. Hashtable每次扩容,新容量为旧容量的2倍加1,而HashMap为旧容量的2倍
  11. Hashtable和HashMap默认负载因子都为0.75
目录
相关文章
|
11天前
|
存储 安全 Java
Java 集合框架中的老炮与新秀:HashTable 和 HashMap 谁更胜一筹?
嗨,大家好,我是技术伙伴小米。今天通过讲故事的方式,详细介绍 Java 中 HashMap 和 HashTable 的区别。从版本、线程安全、null 值支持、性能及迭代器行为等方面对比,帮助你轻松应对面试中的经典问题。HashMap 更高效灵活,适合单线程或需手动处理线程安全的场景;HashTable 较古老,线程安全但性能不佳。现代项目推荐使用 ConcurrentHashMap。关注我的公众号“软件求生”,获取更多技术干货!
33 3
|
10天前
|
人工智能 自然语言处理 Java
FastExcel:开源的 JAVA 解析 Excel 工具,集成 AI 通过自然语言处理 Excel 文件,完全兼容 EasyExcel
FastExcel 是一款基于 Java 的高性能 Excel 处理工具,专注于优化大规模数据处理,提供简洁易用的 API 和流式操作能力,支持从 EasyExcel 无缝迁移。
69 9
FastExcel:开源的 JAVA 解析 Excel 工具,集成 AI 通过自然语言处理 Excel 文件,完全兼容 EasyExcel
|
17天前
|
存储 缓存 Java
Java 并发编程——volatile 关键字解析
本文介绍了Java线程中的`volatile`关键字及其与`synchronized`锁的区别。`volatile`保证了变量的可见性和一定的有序性,但不能保证原子性。它通过内存屏障实现,避免指令重排序,确保线程间数据一致。相比`synchronized`,`volatile`性能更优,适用于简单状态标记和某些特定场景,如单例模式中的双重检查锁定。文中还解释了Java内存模型的基本概念,包括主内存、工作内存及并发编程中的原子性、可见性和有序性。
Java 并发编程——volatile 关键字解析
|
16天前
|
存储 设计模式 算法
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。 行为型模式分为: • 模板方法模式 • 策略模式 • 命令模式 • 职责链模式 • 状态模式 • 观察者模式 • 中介者模式 • 迭代器模式 • 访问者模式 • 备忘录模式 • 解释器模式
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
|
16天前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
结构型模式描述如何将类或对象按某种布局组成更大的结构。它分为类结构型模式和对象结构型模式,前者采用继承机制来组织接口和类,后者釆用组合或聚合来组合对象。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象结构型模式比类结构型模式具有更大的灵活性。 结构型模式分为以下 7 种: • 代理模式 • 适配器模式 • 装饰者模式 • 桥接模式 • 外观模式 • 组合模式 • 享元模式
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
|
16天前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是"将对象的创建与使用分离”。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节。创建型模式分为5种:单例模式、工厂方法模式抽象工厂式、原型模式、建造者模式。
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
|
2天前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
|
15天前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
17天前
|
安全 搜索推荐 数据挖掘
陪玩系统源码开发流程解析,成品陪玩系统源码的优点
我们自主开发的多客陪玩系统源码,整合了市面上主流陪玩APP功能,支持二次开发。该系统适用于线上游戏陪玩、语音视频聊天、心理咨询等场景,提供用户注册管理、陪玩者资料库、预约匹配、实时通讯、支付结算、安全隐私保护、客户服务及数据分析等功能,打造综合性社交平台。随着互联网技术发展,陪玩系统正成为游戏爱好者的新宠,改变游戏体验并带来新的商业模式。

推荐镜像

更多