【Java集合】面试题汇总

  • Java 集合
    • Java 集合概览
    • 1. List, Set, Queue, Map 四者的区别?
    • 2. ArrayList 和 Array(数组)的区别?
    • 3. ArrayList 和 Vector 的区别?
    • 4. Vector 和 Stack 的区别?(了解即可)
    • 5. ArrayList 可以添加 null 值吗?
    • 6. ArrayList 插入和删除元素的时间复杂度?
    • 7. LinkedList 插入和删除元素的时间复杂度?
    • 8. LinkedList 为什么不能实现 RandomAccess 接口?
    • 9. ArrayList 与 LinkedList 区别?
    • 10. 说一说 ArrayList 的扩容机制
    • 11. Comparable 和 Comparator 的区别
    • 12. 比较 HashSet、LinkedHashSet 和 TreeSet 三者的异同
    • 13. Queue 与 Deque 的区别
    • 14. ArrayDeque 与 LinkedList 的区别
    • 15. 说一说 PriorityQueue
    • 16. 什么是 BlockingQueue?
    • 17. BlockingQueue 的实现类有哪些?
    • 18. ArrayBlockingQueue 和 LinkedBlockingQueue 有什么区别?
    • 19. HashMap 和 Hashtable 的区别
    • 20. HashMap 和 HashSet 区别
    • 21. HashMap 和 TreeMap 区别
    • 22. HashSet 如何检查重复?
    • 23. HashMap 的底层实现
    • 24. HashMap 的长度为什么是 2 的幂次方
    • 25. HashMap 多线程操作导致死循环问题
    • 26. HashMap 为什么线程不安全?
    • 27. HashMap 常见的遍历方式?
    • 28. ConcurrentHashMap 和 Hashtable 的区别
    • 29. JDK 1.7 和 JDK 1.8 的 ConcurrentHashMap 实现有什么不同?
    • 30. ConcurrentHashMap 为什么 key 和 value 不能为 null?
    • 31. ConcurrentHashMap 能保证复合操作的原子性吗?
    • 32. ArrayList 扩容机制分析
    • 33. HashMap 扩容机制分析
    • 34. CopyOnWriteArrayList 简介

Java 集合

Java 集合概览

答:

Java 集合,也叫作容器,主要是由两大接口派生而来

  • Collection 接口
  • Map 接口

Java 集合框架如下图所示:

在这里插入图片描述

1. List, Set, Queue, Map 四者的区别?

答:

  • List:存储的元素是有序的、可重复的。
  • Set:存储的元素是无序的、不可重复的。
  • Queue:存储的元素可重复,元素先进先出。
  • Map:使用键值对(key-value)存储元素。key 是无序的、不可重复的

2. ArrayList 和 Array(数组)的区别?

答:

  • ArrayList 支持动态的扩容,Array被创建后不能改变长度了。
  • ArrayList 只能存储对象类型,Array可以存储对象类型和基本数据类型。

3. ArrayList 和 Vector 的区别?

答:

  • ArrayList线程不安全
  • Vector线程安全 。

4. Vector 和 Stack 的区别?(了解即可)

答:

  • Vector 和 Stack 两者都是线程安全的,都是使用 synchronized 关键字进行同步处理。
  • Stack 继承自 Vector,是一个先进后出的栈,而 Vector 是一个列表。

注意: Vector 和 Stack 已经被淘汰,推荐使用并发集合类(例如 ConcurrentHashMapCopyOnWriteArrayList

5. ArrayList 可以添加 null 值吗?

答:

  • 可以

6. ArrayList 插入和删除元素的时间复杂度?

答:

插入:

  • 头部插入:O(n)
  • 尾部插入:如果容量未到极限O(1),到达极限O(n)
  • 指定位置插入:O(n)

删除:

  • 头部删除:O(n)
  • 尾部删除:O(1)
  • 指定位置删除:O(n)

7. LinkedList 插入和删除元素的时间复杂度?

答:

插入:

  • 头部插入:O(1)
  • 尾部插入:O(1)
  • 指定位置插入:O(n)

删除:

  • 头部删除:O(1)
  • 尾部删除:O(1)
  • 指定位置删除:O(n)

8. LinkedList 为什么不能实现 RandomAccess 接口?

答:

  • RandomAccess 是一个标记接口,用来表明实现该接口的类支持随机访问(即可以通过索引快速访问元素)。
  • 由于 LinkedList 底层数据结构是链表,内存地址不连续,只能通过指针来定位,不支持随机快速访问,所以不能实现 RandomAccess 接口

9. ArrayList 与 LinkedList 区别?

答:

  • ArrayList 底层使用的是 Object 数组,因为内存地址连续,可以实现随机访问。ArrayList会预留一定的空间容量。
  • LinkedList 底层使用的是 双向链表(JDK1.6 之前为循环链表),不支持随机访问。LinkedList存储的每个元素要比ArrayList的多(因为要存放后继和前驱节点)。

10. 说一说 ArrayList 的扩容机制

答:

  • 如果调用 ArrayList 的无参构造器的话,默认创建10个大小的容量。
  • 如果添加的元素超过了10,则会创建一个原先容量1.5倍大小的数组。
  • 再将原数组的数据拷贝过去。

11. Comparable 和 Comparator 的区别

答:

  • Comparable 接口和 Comparator 接口都是 Java 中用于排序的接口。
  • Comparable:
    • 要排序的对象实现Comparable 接口,重写 compareTo 方法
  • Comparator:
    • 通常需要使用匿名内部类来实现 Comparator 接口,并重写其中的 compare 方法
Collections.sort(arrayList, new Comparator<Integer>() {@Overridepublic int compare(Integer o1, Integer o2) {return o2.compareTo(o1);}
});

12. 比较 HashSet、LinkedHashSet 和 TreeSet 三者的异同

答:

相同点:

  • 他们都是 set 接口的实现类,都能保证元素的唯一性,都不是线程安全的。

不同点:

底层实现的数据结构不一样。

  • HashSet 底层使用的是 HashMap 实现的。
  • LinkedHashSet 的底层数据结构是链表和哈希表
  • TreeSet 底层是红黑树实现的。

底层实现的数据结构不一样,又会影响他们的使用场景不一样。

  • LinkedHashSet 能够保证插入的元素顺序。
  • TreeSet 支持自定义排序规则

13. Queue 与 Deque 的区别

答:

  • Queue单端队列,只能从一端插入元素,另一端删除元素。
  • Deque双端队列,在队列的两端均可以插入或删除元素。可以模拟栈

14. ArrayDeque 与 LinkedList 的区别

答:

ArrayDequeLinkedList 都实现了 Deque 接口,两者都具有队列的功能

区别:

  • ArrayDeque 是基于可变长的数组和双指针来实现,而 LinkedList 则通过链表来实现。
  • ArrayDeque 不支持存储 NULL 数据,但 LinkedList 支持。
  • ArrayDeque 插入时可能存在扩容LinkedList 插入时需要申请堆空间

15. 说一说 PriorityQueue

答:

PriorityQueue 优先级队列,实现了 Queue 接口。总是优先级最高的元素先出队。

特点:

  • PriorityQueue 利用了二叉堆的数据结构来实现的,底层使用可变长的数组来存储数据。
  • PriorityQueue 插入元素和删除堆顶元素的时间复杂度为 O(logn)
  • PriorityQueue非线程安全的,且不支持存储 NULLnon-comparable 的对象。
  • PriorityQueue 默认是小顶堆,但可以接收一个 Comparator 作为构造参数,从而来自定义元素优先级。

16. 什么是 BlockingQueue?

答:

BlockingQueue (阻塞队列)是一个接口,继承自 Queue

  • 当要取队列中的元素时,如果队列为空,则会阻塞等待,直到有元素为止。
  • 当要将元素放入队列时,如果队列满了,则会阻塞等待,直到有其他元素出队列。

BlockingQueue 常用于生产者-消费者模型中,生产者线程会向队列中添加数据,而消费者线程会从队列中取出数据进行处理。

在这里插入图片描述

17. BlockingQueue 的实现类有哪些?

答:

在这里插入图片描述

Java 中常用的阻塞队列实现类有以下几种:

  • ArrayBlockingQueue:使用数组实现的有界阻塞队列。在创建时需要指定容量大小,并支持公平和非公平两种方式的锁访问机制。
  • LinkedBlockingQueue:使用单向链表实现的可选有界阻塞队列。在创建时可以指定容量大小,如果不指定则默认为Integer.MAX_VALUE。和ArrayBlockingQueue不同的是, 它仅支持非公平的锁访问机制。
  • PriorityBlockingQueue:支持优先级排序的无界阻塞队列。元素必须实现Comparable接口或者在构造函数中传入Comparator对象,并且不能插入 null 元素。
  • SynchronousQueue:同步队列,是一种不存储元素的阻塞队列。每个插入操作都必须等待对应的删除操作,反之删除操作也必须等待插入操作。因此,SynchronousQueue通常用于线程之间的直接传递数据。
  • DelayQueue:延迟队列,其中的元素只有到了其指定的延迟时间,才能够从队列中出队。

18. ArrayBlockingQueue 和 LinkedBlockingQueue 有什么区别?

答:

  • 底层实现:ArrayBlockingQueue 基于数组实现,而 LinkedBlockingQueue 基于链表实现
  • 是否有界:ArrayBlockingQueue 是有界队列,必须在创建时指定容量大小。LinkedBlockingQueue 创建时可以不指定容量大小,默认是Integer.MAX_VALUE,也就是无界的。但也可以指定队列大小,从而成为有界的。
  • 锁是否分离: ArrayBlockingQueue中的锁是没有分离的,即生产和消费用的是同一个锁;LinkedBlockingQueue中的锁是分离的,即生产用的是putLock,消费是takeLock,这样可以防止生产者和消费者线程之间的锁争夺。

19. HashMap 和 Hashtable 的区别

答:

  • HashMap 不是线程安全,Hashtable 线程安全
  • HashMap 的 key 和 value 可以存储 null 值Hashtable 的 key 和 value 不可以存储 null 值
  • JDK8 之后,HashMap的底层数据结构发生了变化(当链表长度大于阈值(默认为 8)时,将链表转化为红黑树(将链表转换成红黑树前会判断,如果当前数组的长度小于 64,那么会选择先进行数组扩容,而不是转换为红黑树)),Hashtable的底层数据结构是数组 + 链表。

20. HashMap 和 HashSet 区别

答:

  • HashSet 的底层实现就是 HashMap
  • HashSet 实例化时,内部其实就是创建了一个 HashMap
  • 调用 HashSet 的 add()方法,就是执行 HashMapput () 方法。

21. HashMap 和 TreeMap 区别

答:

  • TreeMap 实现了 NavigableMap 接口和 SortedMap 接口。
  • 相比于 HashMap ,多了对集合中的元素根据键排序的能力以及对集合内元素的搜索的能力。
  • NavigableMap 接口提供了丰富的方法来搜索和操作键值对,这些方法是基于红黑树的属性实现的,保证了搜索操作的时间复杂度为 O(log n)

22. HashSet 如何检查重复?

答:

  • 当一个元素加入 HashSet 时,首先会根据 hashcode 生成对应的哈希码,确定元素在哈希表中的位置。
  • 当对应位置没有元素时,则会加入到该位置。
  • 如果该位置已经有元素了,则会调用 equals() 方法比较对象的属性是否相同。如果相同则认为是重复元素,则加入失败。
  • 如果对象属性不相同,则会加入到对应位置的链表或是红黑树中。

23. HashMap 的底层实现

答:

哈希流程:

  • HashMap 首先获得 key 的 hashcode ,之后再经过扰动函数(hash函数)处理过后得到 hash 值(再经过hash函数是为了减少碰撞冲突
  • 然后通过 (n - 1) & hash 判断当前元素存放的位置(这里的 n 指的是数组的长度)
  • 如果当前位置存在元素的话,就通过 eaquls() 判断key是否相同。相同的话,直接覆盖,不相同通过拉链法解决冲突。

JDK 8之前

  • HashMap 底层是数组和链表

JDK 8之后

  • 当链表长度大于阈值(默认为 8)(将链表转换成红黑树前会判断,如果当前数组的长度小于 64,那么会选择先进行数组扩容,而不是转换为红黑树)时,将链表转化为红黑树,以减少搜索时间。
  • 为什么使用红黑树?: 红黑树就是为了解决二叉查找树的缺陷,因为二叉查找树在某些情况下会退化成一个线性结构

24. HashMap 的长度为什么是 2 的幂次方

答:

  • 提高运算速度。当长度为 2 的幂次方时, hash%lengthhash&(length-1) 等价,但是位操作效率更高。
  • 减少冲突

25. HashMap 多线程操作导致死循环问题

答:

  • JDK 8之前,HashMap的链表是采取头插法,再多线程情况下可能导致形成环形链表,使查询陷入死循环。
  • JDK 8之后,采用尾插法,避免形成环形链表。
  • 但是在并发环境下,推荐使用 ConcurrentHashMap

26. HashMap 为什么线程不安全?

答:

  • 比如有2个线程同时进行 put 操作。
  • 线程1要插入的元素经过哈希后,确定了要插入的位置。并且此时时间片执行完毕。
  • 线程2获得CPU,线程2同时也要插入对应的位置,线程2插入对应的位置。
  • 线程1获得CPU后,会直接插入到该位置,那么就会覆盖线程2的数据。

27. HashMap 常见的遍历方式?

答:

  • 迭代器(Iterator)方式遍历
  • For Each 方式遍历
  • Streams API 遍历

28. ConcurrentHashMap 和 Hashtable 的区别

答:

实现线程安全的方式不同:

  • Hashtable 使用的是全局锁,任何时刻只能有一个线程进行操作。在并发环境下,效率是非常低。
  • JDK 8之前的 ConcurrentHashMap
    • 分段的数组 + 链表实现的,锁的是每一个段,每个段就相当于是一个Hashtable。
    • 在并发场景下,多线程访问不同的段,就不会存在锁竞争。
  • JDK 8之后的 ConcurrentHashMap
    • Node 数组 + 链表 / 红黑树 实现的,采用 CAS + synchronized 来保证并发安全。
    • synchronized 只锁定当前链表或红黑二叉树的首节点,这样只要 hash 不冲突,就不会产生并发

29. JDK 1.7 和 JDK 1.8 的 ConcurrentHashMap 实现有什么不同?

答:

  • 线程安全实现方式:JDK 1.7 采用 Segment 分段锁来保证安全, Segment 是继承自 ReentrantLock。JDK1.8 采用 Node + CAS + synchronized 保证线程安全,锁粒度更细,synchronized 只锁定当前链表或红黑二叉树的首节点
  • Hash 碰撞解决方法:JDK 1.7 采用拉链法,JDK1.8 采用拉链法结合红黑树。
  • 并发度:JDK 1.7 最大并发度是 Segment 的个数,默认是 16。JDK 1.8 最大并发度是 Node 数组的大小,并发度更大。

30. ConcurrentHashMap 为什么 key 和 value 不能为 null?

答:

  • 避免二义性,如果 key 和 value 可以为null,那么在并发场景下,不知道这个值本身就是 null,还是其他线程修改的null。

31. ConcurrentHashMap 能保证复合操作的原子性吗?

答:

  • 可以使用 ConcurrentHashMap 提供的原子性复合操作。
  • putIfAbsentcomputecomputeIfAbsentcomputeIfPresentmerge

32. ArrayList 扩容机制分析

答:

  • 当添加元素到 ArrayList 时,它会检查当前元素数量是否已经超过了当前容量(内部通过 capacity 属性记录容量),如果超过了,则会进行扩容操作。
  • 当 ArrayList 需要扩容时,它会创建一个新的更大的数组,通常是原来容量的 1.5 倍。然后将旧的元素复制到新的数组中。

33. HashMap 扩容机制分析

答:

  • HashMap 默认的初始化大小为 16。之后每次扩充,容量变为原来的 2 倍
  • HashMap 总是使用 2 的幂作为哈希表的大小。

34. CopyOnWriteArrayList 简介

答:

  • CopyOnWriteArrayList 是并发安全的 List
  • 采用了 写时复制(Copy-On-Write) 的策略
  • 即在进行修改操作(add,set、remove)的时候,先创建底层数组的副本,对副本数组进行修改
  • 修改完之后再将修改后的数组赋值回去,这样就可以保证写操作不会影响读操作了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/808720.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【端云一体化开发】云函数本地运行/调试启动失败的两种解决方案

最近本地调试云函数一直出现这个错误&#xff1a;Before launch task execute failed! details:java.lang.lllegalStateException: npm installfailed 这个问题的原因似乎是运行云函数的时候会重新下载 npm 及相关依赖文件&#xff0c;但是 DevEco 的 npm 模块出错导致这个步骤…

智慧园区平台再升级!智慧迭代,服务升级

伴随物联网、人工智能等技术的迅速发展和智能化水平的提高&#xff0c;智慧园区成为了现代区域经济高质量发展的重要组成部分&#xff0c;上承智慧城市的建设&#xff0c;下接智慧运营和管理。智慧园区是一种基于信息技术的智能化管理模式&#xff0c;通过物联网、大数据、人工…

java中常见的几种排序

常见的几种排序整理 冒泡排序选择排序插入排序希尔排序快速排序归并排序堆排序 冒泡排序 思想&#xff1a;对比当前值的下一个值&#xff0c;如果大就交换位置 代码&#xff1a; /*** 冒泡排序*/ public class bubbleSort {public static void main(String[] args) {int[] ar…

【域适应】深度域适应常用的距离度量函数实现

关于 深度域适应中&#xff0c;有一类方法是实现目标域和源域的特征对齐&#xff0c;特征对齐的衡量函数主要包括MMD&#xff0c;MK-MMD&#xff0c;A-distance&#xff0c;CORAL loss&#xff0c; Wasserstein distance等等。本文总结了常用的特征变换对齐的函数定义。 工具 …

初始C++之缺省参数 函数重载 引用

初始C之缺省参数 函数重载 引用& 文章目录 初始C之缺省参数 函数重载 引用&一、缺省参数1.1 缺省参数的定义1.2 缺省参数的分类1.3 注意事项 二、 函数重载2.1 函数重载的定义2.2 参数个数不同2.3 参数类型不同2.4 类型顺序不同2.5 为什么C语言不支持函数重载 三、引用…

OpenHarmony南向开发案例:【智能保险柜】

样例简介 智能保险柜实时监测保险柜中振动传感器&#xff0c;当有振动产生时及时向用户发出警报。在连接网络后&#xff0c;配合数字管家应用&#xff0c;用户可以远程接收智能保险柜的报警信息。后续可扩展摄像头等设备&#xff0c;实现对危险及时报警&#xff0c;及时处理&a…

探究 ChatGPT 的心脏--Transformer(基础知识第一篇)

Transformer 是 ChatGPT 的核心部分&#xff0c;如果将 AI 看做一辆高速运转的汽车&#xff0c;那么 Transformer 就是最重要的引擎。它是谷歌于 2017 年发表的《Attention is All You Need》中提出的 Sequence-to-sequence 的模型&#xff0c;诞生之后便一统江湖&#xff0c;在…

项目存放在git上,在jenkins使用docker打包并推送到Ubuntu上运行

项目添加dockerfile 在需要打包的工程的根目录添加Dockerfile文件&#xff0c;文件内容&#xff1a; # 设置JAVA版本 FROM openjdk:8 # 指定存储卷&#xff0c;任何向/tmp写入的信息都不会记录到容器存储层 VOLUME /tmp# 拷贝运行JAR包 ARG JAR_FILE COPY ${JAR_FILE} app.jar…

蓝桥杯练习系统(算法训练)ALGO-958 P0704回文数和质数

资源限制 内存限制&#xff1a;256.0MB C/C时间限制&#xff1a;1.0s Java时间限制&#xff1a;3.0s Python时间限制&#xff1a;5.0s 一个数如果从左往右读和从右往左读数字是完全相同的&#xff0c;则称这个数为回文数&#xff0c;比如898,1221,15651都是回文数。编写…

内核驱动更新

1.声明我们是开源的 .c 文件末尾加上 2.在Kconfig里面修改设备&#xff0c;bool&#xff08;双态&#xff09;-----》tristate&#xff08;三态&#xff09; 3.进入menuconfig修改为M 4.编译内核 make modules 也许你会看到一个 .ko 文件 5.复制到根目录文件下 在板子…

4.11作业

服务器端 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include<QTcpServer> //服务器端类 #include<QMessageBox> //消息对话框 #include<QTcpSocket> //客户端类 #include<QList> //链表容器QT_BEGIN_NAMESPACE namespace Ui { cla…

Pycharm远程连接服务器配置详解

背景&#xff1a; 相信很多人都遇到了这种情况&#xff0c;日常的开发和程序的验证都需要在linux环境下验证&#xff0c;而我们都是使用本地windows来进行开发或者脚本的编写&#xff0c;然后再push到远程仓库&#xff0c;再到linux环境下pull下来代码验证&#xff0c;这样每次…

CorelDRAW21.2.4中文最新官方和谐版下载

CorelDRAW是一款由加拿大Corel公司出品的平面设计软件&#xff0c;也被称为CDR。它是一款功能强大的矢量图形制作和排版软件&#xff0c;主要面向绘图设计师和印刷输出人员。该软件提供了矢量插图、页面布局、图片编辑和设计工具&#xff0c;广泛应用于排版印刷、矢量图形编辑及…

HWOD:密码强度等级

一、知识点 回车键的ASCII码是10 如果使用EOF&#xff0c;有些用例不通过 二、题目 1、描述 密码按如下规则进行计分&#xff0c;并根据不同的得分为密码进行安全等级划分。 一、密码长度: 5 分: 小于等于4 个字符 10 分: 5 到7 字符 25 分: 大于等于8 个字符 二、字母: 0…

NotePad++ 快速生成SQL IN (‘’,‘’)

sql In(‘’&#xff0c;‘’)这种形式 第一步&#xff1a;AltC 鼠标放在第一行最左边 第二步 CtrlH $代表行末 第三步 去掉每行换行符 换行可能是"\n" 或者"\r"或者"\r\n" 结果&#xff1a;

容错组合导航

在初始值正确的情况下&#xff0c;惯性导航短期精度较高&#xff0c;但是其误差随着时间是累计的。如果要提高惯性导航的长期精度&#xff0c;就必须提高惯性器件的精度和初始读准精度&#xff0c;这必将大大提高成本。 如果将惯性导航与其他导航系统适当地组合起来&#xff0c…

Java泛型中 T 和 ? 傻傻分不清楚

1.定义&#xff1a; JDK5.0后&#xff0c;Java提供了泛型。 泛型是一种在编译时提供类型安全的方式&#xff0c;允许程序员在定义类、接口和方法时使用类型参数。这样&#xff0c;可以在不损失类型安全的情况下&#xff0c;创建可重用的代码。 泛型有两种主要的使用形式&#x…

linux学习:栈

目录 顺序栈 结构 初始化一个空顺序栈 压栈 出栈 例子 十进制转八进制 链式栈 管理结构体的定义 初始化 压栈 出栈 顺序栈 顺序栈的实现&#xff0c;主要就是定义一块连续的内存来存放这些栈元素&#xff0c;同时为了方便管理&#xff0c; 再定义一个整数变量来代表…

2024中国(宁波)国际宠物用品博览会

2024中国(宁波)国际宠物用品博览会 People&Pet Fair 2024 专注2B交易&#xff0c;关注人宠发展&#xff0c;它经济&#xff0c;势不可挡! 时间&#xff1a;2024年11月14-16日 地点&#xff1a;宁波国际会展中心 详询主办方陆先生 I38&#xff08;前三位&#xff09; …

水离子雾化壁炉与酒店大厅的氛围搭配

将水离子雾化壁炉与酒店大厅的氛围搭配是一个很好的主意&#xff0c;可以为大厅增添舒适、温馨的氛围&#xff0c;以下是一些建议&#xff1a; 迎宾区域&#xff1a;在酒店大厅的迎宾区域设置水离子雾化壁炉&#xff0c;作为客人抵达时的第一印象。壁炉的温馨效果可以让客人感到…