数据结构思维 第九章 `Map`接口

简介: 第九章 Map接口 原文:Chapter 9 The Map interface 译者:飞龙 协议:CC BY-NC-SA 4.0 自豪地采用谷歌翻译在接下来的几个练习中,我介绍了Map接口的几个实现。

第九章 Map接口

原文:Chapter 9 The Map interface

译者:飞龙

协议:CC BY-NC-SA 4.0

自豪地采用谷歌翻译

在接下来的几个练习中,我介绍了Map接口的几个实现。其中一个基于哈希表,这可以说是所发明的最神奇的数据结构。另一个是类似的TreeMap,不是很神奇,但它有附加功能,它可以按顺序迭代元素。

你将有机会实现这些数据结构,然后我们将分析其性能。

但是在我们可以解释哈希表之前,我们将从一个Map开始,它使用键值对的List来简单实现。

9.1 实现MyLinearMap

像往常一样,我提供启动代码,你将填写缺少的方法。这是MyLinearMap类定义的起始:

public class MyLinearMap<K, V> implements Map<K, V> {

    private List<Entry> entries = new ArrayList<Entry>();

该类使用两个类型参数,K是键的类型,V是值的类型。MyLinearMap实现Map,这意味着它必须提供Map接口中的方法。

MyLinearMap对象具有单个实例变量,entries,这是一个EntryArrayList对象。每个Entry都包含一个键值对。这里是定义:

    public class Entry implements Map.Entry<K, V> {
        private K key;
        private V value;

        public Entry(K key, V value) {
            this.key = key;
            this.value = value;
        }

        @Override
        public K getKey() {
            return key;
        }
        @Override
        public V getValue() {
            return value;
        }
    }

Entry没有什么,只是一个键和一个值的容器。该定义内嵌在MyLinearList中,因此它使用相同类型的参数,KV

这就是你做这个练习所需的所有东西,所以让我们开始吧。

9.2 练习 7

在本书的仓库中,你将找到此练习的源文件:

  • MyLinearMap.java包含练习的第一部分的起始代码。
  • MyLinearMapTest.java包含MyLinearMap的单元测试。

你还会找到 Ant 构建文件build.xml

运行ant build来编译源文件。然后运行ant MyLinearMapTest;几个测试应该失败,因为你有一些任务要做。

首先,填写findEntry的主体。这是一个辅助方法,不是Map接口的一部分,但是一旦你让它工作,你可以在几种方法中使用它。给定一个目标键(Key),它应该搜索条目(Entry)并返回包含目标的条目(按照键,而不是值),或者如果不存在则返回null。请注意,我提供了equals,正确比较两个键并处理null

你可以再次运行ant MyLinearMapTest,但即使你的findEntry是正确的,测试也不会通过,因为put不完整。

填充put。你应该阅读Map.put的文档,http://thinkdast.com/listput ,以便你知道应该做什么。你可能希望从一个版本开始,其中put始终添加新条目,并且不会修改现有条目;这样你可以先测试简单的情况。或者如果你更加自信,你可以一次写出整个东西。

一旦你put正常工作,测试containsKey应该通过。

阅读Map.get的文档,http://thinkdast.com/listget ,然后填充方法。再次运行测试。

最后,阅读Map.remove的文档,http://thinkdast.com/maprem 并填充方法。

到了这里,所有的测试都应该通过。恭喜!

9.3 分析MyLinearMap

这一节中,我展示了上一个练习的答案,并分析核心方法的性能。这里是findEntryequals

private Entry findEntry(Object target) {
    for (Entry entry: entries) {
        if (equals(target, entry.getKey())) {
            return entry;
        }
    }
    return null;
}

private boolean equals(Object target, Object obj) {
    if (target == null) {
        return obj == null;
    }
    return target.equals(obj);
}

equals的运行时间可能取决于target键和键的大小 ,但通常不取决于条目的数量,n。那么equals是常数时间。

findEntry中,我们可能会很幸运,并在一开始就找到我们要找的键,但是我们不能指望它。一般来说,我们要搜索的条目数量与n成正比,所以findEntry是线性的。

大部分的MyLinearMap核心方法使用findEntry,包括putget,和remove。这就是他们的样子:

public V put(K key, V value) {
    Entry entry = findEntry(key);
    if (entry == null) {
        entries.add(new Entry(key, value));
        return null;
    } else {
        V oldValue = entry.getValue();
        entry.setValue(value);
        return oldValue;
    }
}
public V get(Object key) {
    Entry entry = findEntry(key);
    if (entry == null) {
        return null;
    }
    return entry.getValue();
}
public V remove(Object key) {
    Entry entry = findEntry(key);
    if (entry == null) {
        return null;
    } else {
        V value = entry.getValue();
        entries.remove(entry);
        return value;
    }
}

put调用findEntry之后,其他一切都是常数时间。记住这个entries是一个ArrayList,所以降魔为添加元素平均是常数时间。如果键已经在映射中,我们不需要添加条目,但我们必须调用entry.getValueentry.setValue,而这些都是常数时间。把它们放在一起,put是线性的。

同样,get也是线性的。

remove稍微复杂一些,因为entries.remove可能需要从一开始或中间删除ArrayList的一个元素,并且需要线性时间。但是没关系:两个线性运算仍然是线性的。

总而言之,核心方法都是线性的,这就是为什么我们将这个实现称为MyLinearMap(嗒嗒!)。

如果我们知道输入的数量很少,这个实现可能会很好,但是我们可以做得更好。实际上,Map所有的核心方法都是常数时间的实现。当你第一次听到这个消息时,可能似乎觉得不可能。实际上我们所说的是,你可以在常数时间内大海捞针,不管海有多大。这是魔法。

我们不是将条目存储在一个大的List中,而是把它们分解成许多短的列表。对于每个键,我们将使用哈希码(在下一节中进行说明)来确定要使用的列表。
使用大量的简短列表比仅仅使用一个更快,但正如我将解释的,它不会改变增长级别;核心功能仍然是线性的。但还有一个技巧:如果我们增加列表的数量来限制每个列表的条目数,就会得到一个恒定时间的映射。你会在下一个练习中看到细节,但是首先要了解哈希!

在下一章中,我将介绍一种解决方案,分析Map核心方法的性能,并引入更有效的实现。

相关文章
|
存储 Java
告别混乱!用Java Map优雅管理你的数据结构
【10月更文挑战第17天】在软件开发中,随着项目复杂度增加,数据结构的组织和管理至关重要。Java中的Map接口提供了一种优雅的解决方案,帮助我们高效、清晰地管理数据。本文通过在线购物平台的案例,展示了Map在商品管理、用户管理和订单管理中的具体应用,有效提升了代码质量和维护性。
263 2
|
存储 Java 开发者
Java Map实战:用HashMap和TreeMap轻松解决复杂数据结构问题!
【10月更文挑战第17天】本文深入探讨了Java中HashMap和TreeMap两种Map类型的特性和应用场景。HashMap基于哈希表实现,支持高效的数据操作且允许键值为null;TreeMap基于红黑树实现,支持自然排序或自定义排序,确保元素有序。文章通过具体示例展示了两者的实战应用,帮助开发者根据实际需求选择合适的数据结构,提高开发效率。
306 2
|
存储 安全 Go
Go语言中的map数据结构是如何实现的?
Go 语言中的 `map` 是基于哈希表实现的键值对数据结构,支持快速查找、插入和删除操作。其原理涉及哈希函数、桶(Bucket)、动态扩容和哈希冲突处理等关键机制,平均时间复杂度为 O(1)。为了确保线程安全,Go 提供了 `sync.Map` 类型,通过分段锁实现并发访问的安全性。示例代码展示了如何使用自定义结构体和切片模拟 `map` 功能,以及如何使用 `sync.Map` 进行线程安全的操作。
392 9
|
存储 Java API
【数据结构】map&set详解
本文详细介绍了Java集合框架中的Set系列和Map系列集合。Set系列包括HashSet(哈希表实现,无序且元素唯一)、LinkedHashSet(保持插入顺序的HashSet)、TreeSet(红黑树实现,自动排序)。Map系列为双列集合,键值一一对应,键不可重复,值可重复。文章还介绍了HashMap、LinkedHashMap、TreeMap的具体实现与应用场景,并提供了面试题示例,如随机链表复制、宝石与石头、前K个高频单词等问题的解决方案。
227 6
【数据结构】map&set详解
|
存储 算法 Java
Set接口及其主要实现类(如HashSet、TreeSet)如何通过特定数据结构和算法确保元素唯一性
Java Set因其“无重复”特性在集合框架中独树一帜。本文解析了Set接口及其主要实现类(如HashSet、TreeSet)如何通过特定数据结构和算法确保元素唯一性,并提供了最佳实践建议,包括选择合适的Set实现类和正确实现自定义对象的hashCode()与equals()方法。
352 4
|
存储 Java 开发者
Java中的Map接口提供了一种优雅的方式来管理数据结构,使代码更加清晰、高效
【10月更文挑战第19天】在软件开发中,随着项目复杂度的增加,数据结构的组织和管理变得至关重要。Java中的Map接口提供了一种优雅的方式来管理数据结构,使代码更加清晰、高效。本文通过在线购物平台的案例,展示了Map在商品管理、用户管理和订单管理中的具体应用,帮助开发者告别混乱,提升代码质量。
170 1
|
数据采集 JavaScript 前端开发
使用 TypeScript 接口优化数据结构
使用 TypeScript 接口优化数据结构
253 11
|
存储 缓存 Java
【用Java学习数据结构系列】HashMap与TreeMap的区别,以及Map与Set的关系
【用Java学习数据结构系列】HashMap与TreeMap的区别,以及Map与Set的关系
182 1
|
存储 自然语言处理 安全
【数据结构】Map的使用与注意事项
【数据结构】Map的使用与注意事项
196 1
|
存储 算法 Java
滚雪球学Java(65):深入理解Java中的Map接口:实现原理剖析
【6月更文挑战第19天】🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
235 3
滚雪球学Java(65):深入理解Java中的Map接口:实现原理剖析

热门文章

最新文章