HashMap,TreeMap,Hashtable,LinkedHashMap的区别

简介: HashMap,TreeMap,Hashtable,LinkedHashMap的区别

HashMap

HashMap的实现原理

  • 基于哈希表(数组+链表+二叉树(红黑树))
  • 默认加载因子为0.75,默认数组大小是16
  • 把对象存储到哈希表中,如何存储?
  • 把key对象通过hash()方法计算hash的值,然后用这个hash值对数组长度取余数(默认是16),来决定对key对象在数组中存储的位置,当这个位置有多个对象时,以链表结构存储,JDK1.8后,当链表长度大于8时,链表转换为红黑树结构存储
  • 这样的目的是为了取值更快,存储的数据量越大,性能的表现越明显
  • 扩充原理:当数组的容量超过75%,数组需要进行扩充,如何扩充
  • 扩充的算法是:当前容量乘以2,扩充次数过多,会影响性能,每次扩充表示哈希表重新散列(重新计算每个位置的存储位置),我们尽量在开发中减少扩充次数带来的性能问题。
  • 线程不安全,适合在单线程里使用

简单使用:

实体类对象:

class Dog {
    private int id;
    private String name;
    public Dog() {
    }
    public Dog(int id, String name) {
        this.id = id;
        this.name = name;
    }
  // get,set,toString
    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()) {
            return false;
        }
        Dog dog = (Dog) o;
        return id == dog.id && Objects.equals(name, dog.name);
    }
    @Override
    public int hashCode() {
        return Objects.hash(id, name);
    }
}
private static void hashMap() {
        Map<Integer, String> map = new HashMap<>();
        map.put(1, "jack");
        map.put(2, "vince");
        map.put(3, "bin");
        map.put(4, "tom");
        map.put(null, null);
        map.put(5, null);
        System.out.println(map.size());
        // 从map遍历1
        map.entrySet().forEach(System.out::println);
        // map遍历2
        Set<Integer> integers = map.keySet();
        for (Integer integer : integers) {
            System.out.println(map.get(integer));
        }
        //map的遍历3(遍历值)
        map.values().forEach(System.out::println);
        // map的遍历4
        map.forEach((integer, s) -> System.out.println(integer + "->" + s));
    }

Hashtable:

  • JDK1.0开始
  • 基于哈希表(数组+链表)
  • 默认数组大小为11,加载因子0.75
  • 扩充方式:原数组大小*2+1
  • 线程安全的,用在多线程访问时

简单使用:

private static void hashtable() {
        Map<String, String> table = new Hashtable<>();
        table.put("one", "lili");
        table.put("two", "lalal");
        table.put("three", "zzz");
        table.forEach((s, s2) -> System.out.println(s + "->" + s2));
    }

TreeMap:

  • 基于二叉树的红黑树实现,方便进行排序

简单应用

private static void treeMap() {
        Map<String, String> map = new LinkedHashMap<>();
        map.put("one", "lili");
        map.put("two", "lalal");
        map.put("three", "zzz");
        map.forEach((s, s2) -> System.out.println(s + "->" + s2));
        Map<Dog, String> dogs = new TreeMap<>(Comparator.comparing(Dog::getId));
        dogs.put(new Dog(1, "1ha"), "dog1");
        dogs.put(new Dog(2, "2ha"), "dog2");
        dogs.put(new Dog(3, "3ha"), "dog3");
        dogs.forEach((s, s2) -> System.out.println(s + "->" + s2));
    }

LinkedHashMap:

  • LinkedHashMap是HashMap的子类,由于HashMap不能保证顺序恒久不变,此类使用一个双重链表来维护元素添加的顺序

简单使用

private static void linkedHashMap() {
        Map<String, String> map = new LinkedHashMap<>();
        map.put("one", "lili");
        map.put("two", "lalal");
        map.put("three", "zzz");
        map.forEach((s, s2) -> System.out.println(s + "->" + s2));
    }


目录
相关文章
|
14天前
|
存储 安全 Java
Java 集合框架中的老炮与新秀:HashTable 和 HashMap 谁更胜一筹?
嗨,大家好,我是技术伙伴小米。今天通过讲故事的方式,详细介绍 Java 中 HashMap 和 HashTable 的区别。从版本、线程安全、null 值支持、性能及迭代器行为等方面对比,帮助你轻松应对面试中的经典问题。HashMap 更高效灵活,适合单线程或需手动处理线程安全的场景;HashTable 较古老,线程安全但性能不佳。现代项目推荐使用 ConcurrentHashMap。关注我的公众号“软件求生”,获取更多技术干货!
34 3
|
2月前
|
安全
HashTable与HashMap的区别
(1)HashTable的每个方法都用synchronized修饰,因此是线程安全的,但同时读写效率很低 (2)HashTable的Key不允许为null (3)HashTable只对key进行一次hash,HashMap进行了两次Hash (4)HashTable底层使用的数组加链表HashTable与HashMap的区别
33 2
|
3月前
|
存储 Java
Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。
【10月更文挑战第19天】本文详细介绍了Java中的HashMap和TreeMap,通过具体示例展示了它们在处理复杂数据结构问题时的应用。HashMap以其高效的插入、查找和删除操作著称,而TreeMap则擅长于保持元素的自然排序或自定义排序,两者各具优势,适用于不同的开发场景。
57 1
|
19天前
|
存储 安全 Java
如何优雅地回答HashSet与HashMap的区别?看这里!
哈喽,大家好!我是小米,29岁程序员。本文聚焦Java开发中经典的面试题——HashSet和HashMap的区别。HashSet基于HashMap实现,存储唯一值;HashMap存储键值对。两者在数据结构、使用场景、操作方法等方面有显著差异。HashSet无序且依赖元素的hashCode和equals方法保证唯一性,而HashMap需注意线程安全问题。掌握这些知识点,助你轻松应对面试。更多技术干货,欢迎关注我的微信公众号“软件求生”。
33 4
|
3月前
|
存储 Java API
详细解析HashMap、TreeMap、LinkedHashMap等实现类,帮助您更好地理解和应用Java Map。
【10月更文挑战第19天】深入剖析Java Map:不仅是高效存储键值对的数据结构,更是展现设计艺术的典范。本文从基本概念、设计艺术和使用技巧三个方面,详细解析HashMap、TreeMap、LinkedHashMap等实现类,帮助您更好地理解和应用Java Map。
83 3
|
3月前
|
存储 缓存 安全
在Java的Map家族中,HashMap和TreeMap各具特色
【10月更文挑战第19天】在Java的Map家族中,HashMap和TreeMap各具特色。HashMap基于哈希表实现,提供O(1)时间复杂度的高效操作,适合性能要求高的场景;TreeMap基于红黑树,提供O(log n)时间复杂度的有序操作,适合需要排序和范围查询的场景。两者在不同需求下各有优势,选择时需根据具体应用场景权衡。
43 2
|
3月前
|
Java
让星星⭐月亮告诉你,HashMap中保证红黑树根节点一定是对应链表头节点moveRootToFront()方法源码解读
当红黑树的根节点不是其对应链表的头节点时,通过调整指针的方式将其移动至链表头部。具体步骤包括:从链表中移除根节点,更新根节点及其前后节点的指针,确保根节点成为新的头节点,并保持链表结构的完整性。此过程在Java的`HashMap$TreeNode.moveRootToFront()`方法中实现,确保了高效的数据访问与管理。
34 2
|
3月前
|
Java 索引
让星星⭐月亮告诉你,HashMap之往红黑树添加元素-putTreeVal方法源码解读
本文详细解析了Java `HashMap` 中 `putTreeVal` 方法的源码,该方法用于在红黑树中添加元素。当数组索引位置已存在红黑树类型的元素时,会调用此方法。具体步骤包括:从根节点开始遍历红黑树,找到合适位置插入新元素,调整节点指针,保持红黑树平衡,并确保根节点是链表头节点。通过源码解析,帮助读者深入理解 `HashMap` 的内部实现机制。
48 2
|
3月前
|
算法 Java 容器
Map - HashSet & HashMap 源码解析
Map - HashSet & HashMap 源码解析
69 0
|
16天前
|
存储 缓存 Java
HashMap源码剖析-put流程
更好地掌握 `HashMap` 的内部实现原理,提高编写高效代码的能力。掌握这些原理不仅有助于优化性能,还可以帮助解决实际开发中的问题。
45 13