原文: https://howtodoinjava.com/interview-questions/useful-java-collection-interview-questions/

毫无疑问,java 集合是最重要的领域之一,无论您是初级还是高级,您都可以在任何位置对其进行测试。 范围如此之广,几乎不可能涵盖所有问题。 但是,根据我以前的面试,我尝试提出您必须知道的尽可能多的优秀的 java 集合面试问题

我的目标是初学者和高级问题,所以如果您发现一些基本问题,请多多包涵,因为它们对某些初级开发人员可能有用。

  1. Java collection interview questions
  2. General questions
  3. 1) What is the Java Collections API? List down its advantages?
  4. 2) Explain Collections hierarchy?
  5. 3) Why Collection interface does not extend Cloneable and Serializable interface?
  6. 4) Why Map interface does not extend Collection interface?
  7. List interface related
  8. 5) Why we use List interface? What are main classes implementing List interface?
  9. 6) How to convert an array of String to ArrayList?
  10. 7) How to reverse the list?
  11. Set interface related
  12. 8) Why we use Set interface? What are main classes implementing Set interface?
  13. 9) How HashSet store elements?
  14. 10) Can a null element added to a TreeSet or HashSet?
  15. Map interface related
  16. 11) Why we use Map interface? What are main classes implementing Map interface?
  17. 12) What are IdentityHashMap and WeakHashMap?
  18. 13) Explain ConcurrentHashMap? How it works?
  19. 14) How hashmap works?
  20. 15) How to design a good key for hashmap?
  21. 16) What are different Collection views provided by Map interface?
  22. 17) When to use HashMap or TreeMap?
  23. Tell the difference questions
  24. 18) Difference between Set and List?
  25. 19) Difference between List and Map?
  26. 20) Difference between HashMap and HashTable?
  27. 21) Difference between Vector and ArrayList?
  28. 22) Difference between Iterator and Enumeration?
  29. 23) Difference between HashMap and HashSet?
  30. 24) Difference between Iterator and ListIterator?
  31. 25) Difference between TreeSet and SortedSet?
  32. 26) Difference between ArrayList and LinkedList?
  33. More questions
  34. 27) How to make a collection read only?
  35. 28) How to make a collection thread safe?
  36. 29) Why there is not method like Iterator.add() to add elements to the collection?
  37. 30) What are different ways to iterate over a list?
  38. 31) What do you understand by iterator fail-fast property?
  39. 32) What is difference between fail-fast and fail-safe?
  40. 33) How to avoid ConcurrentModificationException while iterating a collection?
  41. 34) What is UnsupportedOperationException?
  42. 35) Which collection classes provide random access of its elements?
  43. 36) What is BlockingQueue?
  44. 37) What is Queue and Stack, list their differences?
  45. 38) What is Comparable and Comparator interface?
  46. 39) What are Collections and Arrays class?
  47. 40) Recommended resources

不要浪费时间,让我们深入研究 Java 集合的概念。

Java 集合面试一般问题

1)什么是 Java 集合框架? 列出其优势?

根据定义,集合是代表一组对象的对象。 像集合论一样,集合是一组元素。 很简单!

在 JDK 1.2 之前,JDK 具有一些工具类,例如VectorHashTable,但是没有集合框架的概念。 从 JDK 1.2 以后,JDK 感到需要对可重用数据结构提供一致的支持。 最后,集合框架主要由 Joshua Bloch 设计和开发,并在 JDK 1.2 中引入。

Java 集合的最明显的优点可以列出为

  • 随时可用的代码,减少了编程工作
  • 由于数据结构和算法的高性能实现而提高了性能
  • 通过建立公共语言来回传递集合,从而在不相关的 API 之间提供互操作性
  • 通过仅学习一些顶级接口和受支持的操作,易于学习的 API

2)解释集合的层次结构?

Java 面试的 40 个热门问答集 - 图1

Java 集合的层次结构

如上图所示,集合框架顶部有一个接口,即集合。 通过集,列表和队列接口对其进行了扩展。 然后在这 3 个分支中还有其他类别的负载,我们将在以下问题中学习。

记住Collection接口的签名。 它会在很多问题上帮助您。

  1. public interface Collection extends Iterable {
  2. //method definitions
  3. }

框架还包含Map接口,它是集合框架的一部分。 但它不会扩展Collection接口。 我们将在此问题库中的第四个问题中看到原因。

3)为什么Collection接口没有扩展CloneableSerializable接口?

好吧,最简单的答案是“不需要这样做”。 扩展接口仅表示您正在创建接口的子类型,换句话说,不希望使用更专门的行为和Collection接口来实现CloneableSerializable接口。

另一个原因是并非每个人都有理由拥有Cloneable集合,因为如果它具有非常大的数据,那么每个不必要的克隆操作都将消耗大量内存。 初学者可能在不知道后果的情况下使用它。

另一个原因是CloneableSerializable是非常专门的行为,因此仅在需要时才应实现。 例如,集合中的许多具体类都实现了这些接口。 因此,如果您想要此功能。 使用这些集合类,否则使用其替代类。

4)为什么Map接口没有扩展Collection接口?

这个面试问题的一个很好的答案是“因为它们不兼容”。 集合具有方法add(Object o)。 Map 无法使用这种方法,因为它需要键值对。 还有其他原因,例如Map支持keySetvalueSet等。Collection类没有此类视图。

由于存在如此大的差异,因此在Map接口中未使用Collection接口,而是在单独的层次结构中构建。

Java 集合面试 – 列出接口问题

5)为什么要使用List接口? 什么是实现List接口的主要类?

Java 列表是元素的“有序”集合。 该排序是基于零的索引。 它不关心重复项。 除了在Collection接口中定义的方法外,它确实有自己的方法,它们在很大程度上也要根据元素的索引位置来操作集合。 这些方法可以分为搜索,获取,迭代和范围视图。 以上所有操作均支持索引位置。

实现List接口的主要类为: StackVectorArrayListLinkedList。 在 Java 文档中阅读有关它们的更多信息。

6)如何将String数组转换为arraylist

这更多是一个程序性问题,在初学者水平上可以看到。 目的是检查集合工具类中申请人的知识。 现在,让我们了解Collection框架中有两个工具类,它们大多数在面试中看到,即CollectionsArrays

集合类提供了一些静态函数来对集合类型执行特定操作。 数组提供了要在数组类型上执行的工具函数。

  1. //String array
  2. String[] words = {"ace", "boom", "crew", "dog", "eon"};
  3. //Use Arrays utility class
  4. List wordList = Arrays.asList(words);
  5. //Now you can iterate over the list

请注意,此函数并非特定于String类,它将返回数组属于任何类型的元素的List。 例如:

  1. //String array
  2. Integer[] nums = {1,2,3,4};
  3. //Use Arrays utility class
  4. List numsList = Arrays.asList(nums);

7)如何反转列表?

这个问题就像上面的测试您对集合工具类的了解。 使用它的reverse()方法可以反转列表。

  1. Collections.reverse(list);

Java 集合面试 – `Set``接口问题

8)为什么要使用Set接口? 什么是实现Set接口的主要类?

对集合论中的数学集合进行建模。 Set接口类似于List接口,但有一些区别。 首先,它是未排序的集合。 因此,添加或删除元素时不会保留任何顺序。 它提供的主要功能是“元素的唯一性”。 它不支持重复元素。

Set还对equalshashCode操作的行为增加了更强的约定,从而即使它们的实现类型不同,也可以有意义地比较Set实例。 如果两个Set实例包含相同的元素,则它们相等。

基于上述原因,它没有基于列表之类的元素的索引进行的操作。 它只有由Collection接口继承的方法。

实现Set接口的主要类为:EnumSetHashSetLinkedHashSetTreeSet。 阅读更多有关 Java 文档的信息。

9)HashSet如何存储元素?

您必须知道HashMap存储具有一个条件的键值对,即键将是唯一的。 HashSet使用映射的此功能来确保元素的唯一性。 在HashSet类中,映射声明如下:

  1. private transient HashMap<E,Object> map;
  2. //This is added as value for each key
  3. private static final Object PRESENT = new Object();

因此,将元素存储在HashSet中时,会将元素存储为映射中的键,将“PRESENT”对象存储为值。 (请参见上面的声明)。

  1. public boolean add(E e) {
  2. return map.put(e, PRESENT)==null;
  3. }

我强烈建议您阅读这篇文章:HashMap如何在 Java 中工作?这篇文章将帮助您轻松地回答所有与HashMap相关的问题。

10)是否可以将null元素添加到TreeSet或 HashSet 中?

如您所见,上一个问题的add()方法中没有null检查。 并且HashMap还允许一个null键,因此在HashSet中允许一个“null”。

TreeSet使用与HashSet相同的概念进行内部逻辑,但是使用NavigableMap来存储元素。

  1. private transient NavigableMap<E,Object> m;
  2. // Dummy value to associate with an Object in the backing Map
  3. private static final Object PRESENT = new Object();

NavigableMapSortedMap的子类型,不允许使用null键。 因此,本质上,TreeSet也不支持空键。 如果您尝试在TreeSet中添加null元素,它将抛出NullPointerException

Java 集合面试 – Map接口问题

11)为什么要使用Map接口? 什么是实现Map接口的主要类?

Map接口是一种特殊的集合类型,它用于存储键值对。 因此,它不会扩展Collection接口。 该接口提供了在映射的各种视图上添加,删除,搜索或迭代的方法。

实现Map接口的主要类有:HashMapHashtableEnumMapIdentityHashMapLinkedHashMapProperties

12)什么是 IdentityHashMapWeakHashMap

IdentityHashMapHashMap相似,不同之处在于在比较元素时使用引用相等性。 IdentityHashMap类不是一种广泛使用的Map实现。 尽管此类实现了Map接口,但它有意违反Map的一般协定,该协定要求在比较对象时必须使用equals()方法。 IdentityHashMap设计为仅在少数情况下使用,其中需要引用相等语义。

WeakHashMapMap接口的实现,该接口仅存储对其键的弱引用。 当不再在WeakHashMap外部引用键值对时,仅存储弱引用将允许对键值对进行垃圾回收。 该类主要用于与equals方法使用==运算符测试对象标识的键对象一起使用。 一旦丢弃了这样的键,就永远无法重新创建它,因此以后不可能在WeakHashMap中对该键进行查找,并且会惊讶地发现它的项目已被删除。

13)解释ConcurrentHashMap吗? 怎么运行的?

来自 Java 文档:

支持检索的完全并发和用于更新的可调整预期并发的哈希表。 此类遵循与Hashtable相同的功能规范,并且包括与Hashtable的每个方法相对应的方法的版本。 但是,即使所有操作都是线程安全的,检索操作也不需要进行锁定,并且不支持以阻止所有访问的方式锁定整个表。 在依赖于其线程安全性但不依赖于其同步详细信息的程序中,此类可与Hashtable完全互操作。

阅读有关ConcurrentHashMap面试问题的更多信息。

14)HashMap如何工作?

最重要的问题在每个工作面试中最有可能出现。 您必须在这个主题上非常清楚。不仅因为它是最常被问到的问题,而且会让您对与集合 API 相关的其他问题打开思路。

这个问题的答案非常大,您应该阅读我的文章:HashMap如何工作? 现在,让我们记住HashMap在哈希原理上工作。 根据定义,映射是:“将键映射到值的对象”。 为了存储这种结构,使用了内部类Entry

  1. static class Entry implements Map.Entry
  2. {
  3. final K key;
  4. V value;
  5. Entry next;
  6. final int hash;
  7. ...//More code goes here
  8. }

此处,键和值变量用于存储键值对。 整个项目对象存储在数组中。

  1. /**
  2. * The table, re-sized as necessary. Length MUST Always be a power of two.
  3. */
  4. transient Entry[] table;

数组的索引是根据Key对象的哈希码计算的。 阅读更多链接主题。

15)如何为哈希表设计一个好的键?

在回答HashMap如何工作后,通常会跟进另一个好问题。 好吧,最重要的约束是,您将来必须能够取回值对象。 否则,没有使用这种数据结构。 如果您了解hashmap的工作原理,将会发现它很大程度上取决于Key对象的hashCode()equals()方法。

因此,好的键对象必须一次又一次提供相同的hashCode(),无论它被获取了多少次。 同样,与equals()方法比较时,相同的键必须返回true,而不同的键必须返回false

因此,不变类被认为是HashMap的最佳候选者。

阅读更多:如何为HashMap设计一个好的键?

16)Map接口提供哪些不同的集合视图?

Map接口提供了 3 个存储在其中的键值对的视图:

  • 键集视图
  • 值集视图
  • 项集视图

可以使用迭代器浏览所有视图。

17)什么时候使用HashMapTreeMap

HashMap是众所周知的类,我们所有人都知道。 因此,我将离开这部分,说它用于存储键值对,并允许对这样的对集合执行许多操作。

TreeMapHashMap的特殊形式。 它维护HashMap类中缺少的键的顺序。 默认情况下,此排序为“自然排序”。 通过提供Comparator类的实例,可以覆盖默认顺序,该类的compare方法将用于维护键的顺序。

请注意,所有插入映射的键都必须实现Comparable接口(这是确定顺序的必要条件)。 此外,所有这些键必须相互可比较:k1.compareTo(k2)不得为映射中的任何键k1k2抛出ClassCastException。 如果用户尝试将键放入违反此约束的映射中(例如,用户尝试将字符串键放入其键为整数的映射中),则put(Object key, Object value)调用将引发ClassCastException

Java 集合面试 – 讲述差异问题

18)SetList之间的区别?

最明显的区别是:

  • Set是无序集合,其中List是基于零索引的有序集合。
  • 列表允许重复元素,但Set不允许重复。
  • List不会阻止插入空元素(随您喜欢),但是Set将只允许一个空元素。

19)列表和映射之间的区别?

也许是最简单的问题。 列表是元素的集合,而映射是键值对的集合。 实际上,有很多差异源自第一个陈述。 它们具有单独的顶层接口,单独的一组通用方法,不同的受支持方法和不同的集合视图

我会花很多时间来回答这个问题,仅作为第一个区别就足够了。

20)HashMapHashTable之间的区别?

Java 中的HashMapHashtable之间有一些区别:

  • Hashtable是同步的,而HashMap不是同步的。
  • 哈希表不允许使用空键或空值。 HashMap允许一个空键和任意数量的空值。
  • HashMapHashtable之间的第三个重要区别是HashMap中的Iterator是快速失败的迭代器,而Hashtable的枚举器则不是。

21)VectorArrayList之间的区别?

让我们记下差异:

  • Vector的所有方法都是同步的。 但是,ArrayList的方法不同步。
  • Vector是在 JDK 的第一个版本中添加的旧类。 当在 Java 中引入集合框架时,ArrayList是 JDK 1.2 的一部分。
  • 默认情况下,Vector在内部调整大小时会将其数组的大小加倍。 但是,重新调整大小时,ArrayList的大小增加一半。

22)迭代器和枚举器之间的区别?

迭代器与枚举在以下三个方面有所不同:

  • 迭代器允许调用方使用其remove()方法在迭代过程中从基础集合中删除元素。 使用枚举器时,不能从集合中添加/删除元素。
  • 枚举器在旧类(例如Vector/Stack等)中可用,而Iterator在所有现代集合类中可用。
  • 另一个小的区别是Iterator改进了方法名称,例如Enumeration.hasMoreElement()变为Iterator.hasNext()Enumeration.nextElement()变为Iterator.next()等。

23)HashMapHashSet之间的区别?

HashMap是键值对的集合,而HashSet是唯一元素的无序集合。 而已。 无需进一步描述。

24)IteratorListIterator之间的区别?

有三个区别:

  • 我们可以使用Iterator遍历SetList以及MapObject类型。 但是列表迭代器可用于遍历列表类型的对象,但不能遍历对象的集合类型。
  • 通过使用Iterator,我们只能从正向检索Collection对象中的元素,而List Iterator则允许您使用hasPrevious()previous()方法在任一方向上遍历。
  • ListIterator允许您使用add() remove()方法修改列表。 使用Iterator不能添加,只能删除元素。

25)TreeSetSortedSet之间的区别?

SortedSetTreeSet实现的接口。 就是这样!

26)ArrayListLinkedList之间的区别?

  • LinkedList将元素存储在双链列表数据结构中。 ArrayList将元素存储在动态调整大小的数组中。
  • LinkedList允许进行固定时间的插入或删除,但只允许顺序访问元素。 换句话说,您可以向前或向后浏览列表,但是在中间抓取一个元素所花费的时间与列表的大小成正比。 另一方面,ArrayList允许随机访问,因此您可以在固定时间内抓取任何元素。 但是,从末端开始的任何地方添加或删除,都需要将后面的所有元素移开,以形成开口或填补空白。
  • LinkedListArrayList具有更多的内存开销,因为在ArrayList中,每个索引仅保存实际的对象(数据),但是在LinkedList的情况下,每个节点都保存下一个和上一个节点的数据以及地址。

更多面试面试问题

27)如何使集合只读?

使用以下方法:

  • Collections.unmodifiableList(list);
  • Collections.unmodifiableSet(set);
  • Collections.unmodifiableMap(map);

这些方法采用集合参数,并返回一个具有与原始集合中相同的元素的新的只读集合。

28)如何使集合线程安全?

使用以下方法:

  • Collections.synchronizedList(list);
  • Collections.synchronizedSet(set);
  • Collections.synchronizedMap(map);

上面的方法将集合作为参数并返回相同类型的集合,这些类型是同步的且线程安全的。

29)为什么没有像Iterator.add()这样的方法将元素添加到集合中?

迭代器的唯一目的是通过集合进行枚举。 所有集合都包含add()方法以实现您的目的。 添加Iterator毫无意义,因为集合可能有序,也可能没有排序。 而且add()方法对于有序和无序集合不能具有相同的实现。

30)有哪些不同的方法可以遍历列表?

您可以使用以下方式遍历列表:

  • 迭代器循环
  • for循环
  • for循环(高级)
  • While循环

阅读更多: http://www.mkyong.com/java/how-do-loop-iterate-a-list-in-java/

31)通过迭代器快速失败属性您了解什么?

快速失败迭代器一旦意识到自迭代开始以来就已更改集合的结构,便会失败。 结构更改意味着在一个线程迭代该集合时,从集合中添加,删除或更新任何元素。

通过保留修改计数来实现快速失败行为,如果迭代线程实现了修改计数的更改,则会引发ConcurrentModificationException

32)快速失败和故障安全之间有什么区别?

您已经在上一个问题中理解了快速失败。 故障安全迭代器与快速失败相反。 如果您修改要在其上进行迭代的基础集合,它们将永远不会失败,因为它们在集合的克隆而不是原始集合上起作用,这就是为什么将它们称为故障保护迭代器。

CopyOnWriteArrayList的迭代器是故障安全迭代器的示例,而且ConcurrentHashMap keySet编写的迭代器也是故障安全迭代器,并且永远不会抛出ConcurrentModificationException

33)如何在迭代集合时避免ConcurrentModificationException

您应该首先尝试查找故障安全的另一个替代迭代器。 例如,如果您正在使用List,则可以使用ListIterator。 如果它是旧式集合,则可以使用枚举。

如果上述选项不可行,则可以使用以下三种更改之一:

  • 如果使用的是 JDK1.5 或更高版本,则可以使用ConcurrentHashMapCopyOnWriteArrayList类。 这是推荐的方法。
  • 您可以将列表转换为数组,然后在数组上进行迭代。
  • 您可以通过将列表放在同步块中来在迭代时锁定列表。

请注意,最后两种方法会导致性能下降。

34)什么是UnsupportedOperationException

实际的集合类型不支持的被调用方法抛出的异常

例如,如果您使用“Collections.unmodifiableList(list)”创建一个只读列表列表,然后调用add()remove()方法,那将会发生什么。 它应该明确抛出UnsupportedOperationException

35)哪些集合类可随机访问其元素?

ArrayListHashMapTreeMapHashtable类提供对其元素的随机访问。

36)什么是BlockingQueue

一个队列,它另外支持以下操作:在检索元素时等待队列变为非空,并在存储元素时等待队列中的空间变为可用。

BlockingQueue方法有四种形式:一种抛出异常,第二种返回一个特殊值(根据操作的不同,返回nullfalse),第三种无限期地阻塞当前线程,直到操作成功为止;第四种仅在放弃之前给出最大时间限制。

阅读文章中的阻塞队列示例用法: 如何使用阻塞队列?

37)什么是队列和栈,列出它们之间的差异?

设计用于在处理之前保存元素的集合。 除了基本的集合操作之外,队列还提供其他插入,提取和检查操作。

通常但不一定以 FIFO(先进先出)的方式对元素进行排序。

栈也是队列的一种形式,但有一个区别,那就是 LIFO(后进先出)。

无论使用哪种顺序,队列的开头都是该元素,可以通过调用remove()poll()将其删除。 另请注意,栈和向量都已同步。

用法:如果要按接收顺序处理传入流,请使用队列。适用于工作列表和处理请求。
如果只想从栈顶部推动并弹出,请使用栈。 适用于递归算法。

38)什么是ComparableComparator接口?

在 Java 中。 所有具有自动排序功能的集合都使用比较方法来确保元素的正确排序。 例如,使用排序的类为TreeSetTreeMap等。

为了对一个类的数据元素进行排序,需要实现ComparatorComparable接口。 这就是所有包装器类(例如IntegerDoubleString类)都实现Comparable接口的原因。

Comparable帮助保留默认的自然排序,而Comparator帮助以某些特殊的必需排序模式对元素进行排序。 比较器的实例,通常在支持集合时作为集合的构造器参数传递。

39)什么是CollectionsArrays类?

CollectionsArrays类是支持集合框架核心类的特殊工具类。 它们提供工具函数以获取只读/同步集合,以各种方式对集合进行排序等。

数组还帮助对象数组转换为集合对象。 数组还具有一些函数,有助于复制或处理部分数组对象。

40)推荐资源

好吧,这不是面试的问题.. :-)。 这只是为了好玩。 但是您应该真正阅读我的博客,以获取有关集合框架知识的更多帖子。

希望这些 Java 集合面试问题对您的下一次面试有所帮助。 此外,除了本文之外,我建议您阅读更多有关上述问题的信息。 更多的知识只会帮助您。

学习愉快!