【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 的扩容机制

答:

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.rhkb.cn/news/301215.html

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

相关文章

paddle实现手写数字模型(一)

参考文档&#xff1a;paddle官网文档调试代码如下&#xff1a; LeNet.py import paddle import paddle.nn.functional as Fclass LeNet(paddle.nn.Layer):def __init__(self):super().__init__()self.conv1 paddle.nn.Conv2D(in_channels1,out_channels6,kernel_size5,stride…

YOLOV9 + 双目测距

YOLOV9 双目测距 1. 环境配置2. 测距流程和原理2.1 测距流程2.2 测距原理 3. 代码部分解析3.1 相机参数stereoconfig.py3.2 测距部分3.3 主代码yolov9-stereo.py 4. 实验结果4.1 测距4.2 视频展示 相关文章 1. YOLOV5 双目测距&#xff08;python&#xff09; 2. YOLOv7双目…

强化学习MPC——(一)

1.什么是强化学习 强化学习是机器学习的一种&#xff0c;是一种介于监督学习和非监督学习的机器学习方法。 学习二字就很形象的说明了这是一种利用数据&#xff08;任何形式的&#xff09;来实现一些已有问题的方法&#xff0c;学习方法&#xff0c;大致可以分为机器学习&…

说说TCP为什么需要三次握手和四次挥手?

一、三次握手 三次握手&#xff08;Three-way Handshake&#xff09;其实就是指建立一个TCP连接时&#xff0c;需要客户端和服务器总共发送3个包 主要作用就是为了确认双方的接收能力和发送能力是否正常、指定自己的初始化序列号为后面的可靠性传送做准备 过程如下&#xff…

Redis 常见面试题

目录 1. Redis是什么&#xff1f;2. Redis优缺点&#xff1f;3. Redis为什么这么快&#xff1f;4. 既然Redis那么快&#xff0c;为什么不用它做主数据库&#xff0c;只用它做缓存&#xff1f;5. Redis的线程模型&#xff1f;6. Redis 采用单线程为什么还这么快&#xff1f;7. R…

如何使用生成式人工智能撰写关于新产品发布的文章?

利用生成式人工智能撰写新产品发布文章确实是一种既有创意又高效的内容生成方式。以下是如何做到这一点的指南&#xff0c;附带一些背景信息&#xff1a; • 背景&#xff1a;在撰写文章之前&#xff0c;收集有关您的新产品的信息。这包括产品的名称、类别、特点、优势、目标受…

解决Xshell连接不上虚拟机

相信有很多同学和我一样遇到这个问题&#xff0c;在网上看了很多教程基本上都先让在虚拟机输入ifconfig命令查看ip地址&#xff0c;弄来弄去最后还是解决不了&#x1f62d;&#x1f62d;&#xff0c;其实问题根本就不在命令上&#xff0c;很大概率是我们的虚拟机没有开启网卡&a…

基于单片机便携式测振仪的研制系统设计

**单片机设计介绍&#xff0c;基于单片机便携式测振仪的研制系统设计 文章目录 一 概要二、功能设计三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机便携式测振仪的研制系统设计概要主要涉及利用单片机作为核心控制器件&#xff0c;结合测振原理和技术&#x…

python-可视化篇-turtle-画爱心

文章目录 原效果替换关键字5为8&#xff0c;看看效果改下颜色 原效果 import turtle as tt.color(red,pink) t.begin_fill() t.width(5) t.left(135) t.fd(100) t.right(180) t.circle(50,-180) t.left(90) t.circle(50,-180) t.right(180) t.fd(100) t.pu() t.goto(50,-30) t…

蓝鲸6.1 CMDB 事件推送的开源替代方案

本文来自腾讯蓝鲸智云社区用户&#xff1a;木讷大叔爱运维 背景 在蓝鲸社区“社区问答”帖子中发现这么一个需求&#xff1a; 究其原因&#xff0c;我在《不是CMDB筑高墙&#xff0c;运维需要一定的开发能力&#xff01;》一文中已经介绍&#xff0c;在此我再简单重复下&#…

JavaScript实现全选、反选功能(Vue全选、反选,js原生全选、反选)

简介&#xff1a; 在JavaScript中&#xff0c;实现全选和反选通常是通过操作DOM元素和事件监听来实现&#xff1b; 全选功能&#xff1a;当用户点击一个“全选”复选框时&#xff0c;页面中所有具有相同类名的复选框都将被选中&#xff1b; 反选功能&#xff1a;用户点击一个…

ARP寻址过程

当知道目标的IP但是不知道目标的Mac地址的时候就需要借助ARP寻址获取目标的Mac地址&#xff0c;传输层借助四元组&#xff08;源IP源端口&#xff1a;目标IP目标端口&#xff09;匹配&#xff0c;网络层借助IP匹配&#xff0c;数据链路层则根据Mac地址匹配&#xff0c;数据传输…

局域网共享文件夹怎么加密?局域网共享文件夹加密方法介绍

在企业局域网中&#xff0c;共享文件夹扮演着重要的角色。为了保护数据安全&#xff0c;我们需要加密保护局域网共享文件夹。那么&#xff0c;局域网共享文件夹怎么加密&#xff1f;下面我们来了解一下吧。 局域网共享文件夹加密方法 局域网共享文件夹加密推荐使用共享文件夹加…

在git上先新建仓库-把本地文件提交远程

一.在git新建远程项目库 1.选择新建仓库 以下以gitee为例 2.输入仓库名称&#xff0c;点击创建 这个可以选择仓库私有化还公开权限 3.获取仓库clone链接 这里选择https模式就行&#xff0c;就不需要配置对电脑进行sshkey配置了。只是需要每次提交输入账号密码 二、远…

万字源码解析!彻底搞懂 HashMap【一】:概念辨析与构造方法源码解析

HashMap 的底层原理和扩容机制一直都是面试的时候经常被问到的问题&#xff0c;同时也是集合源码中最难阅读的一部分&#x1f622;&#xff0c;之前更新的 ArrayList 源码阅读收获了很多朋友的喜欢&#xff0c;也给了我很多自信&#xff1b;本次我准备完成一个关于 HashMap 源码…

python练习三

模式A num int(input("请输入模式A的层数&#xff1a;")) for i in range(1, num 1):# 画数字for j in range(1, i 1):print(str(j) "\t", end"")print() 模式B num int(input("请输入模式B的层数&#xff1a;")) for i in ran…

九州金榜|孩子叛逆的原因是什么?

孩子随着年龄增长都会出现叛逆心理&#xff0c;很多家长不知道孩子为什么会出现叛逆心理&#xff0c;也不知道如何去引导孩子&#xff0c;下面九州金榜家庭教育就带大家了解一下孩子出现叛逆的原因。 一、心理需求增加 孩子对新事物的探索以及追求会随着人际交往扩大而增加&am…

2024年MathorCup妈妈杯数学建模思路D题思路解析+参考成品

1 赛题思路 (赛题出来以后第一时间在群内分享&#xff0c;点击下方群名片即可加群) 2 比赛日期和时间 报名截止时间&#xff1a;2024年4月11日&#xff08;周四&#xff09;12:00 比赛开始时间&#xff1a;2024年4月12日&#xff08;周五&#xff09;8:00 比赛结束时间&…

买卖股票的最佳时机IV

题目链接 买卖股票的最佳时机 IV 题目描述 注意点 1 < k < 1001 < prices.length < 10000 < prices[i] < 1000不能同时参与多笔交易&#xff08;必须在再次购买前出售掉之前的股票&#xff09;最多可以完成 k 笔交易 解答思路 本题与买卖股票的最佳时机…

单例模式--理解

单例模式 单例模式是指在内存中只会创建且仅创建一次对象的设计模式。在程序中多次使用同一个对象且作用相同时&#xff0c;为了防止频繁地创建对象使得内存飙升&#xff0c;单例模式可以让程序仅在内存中创建一个对象&#xff0c;让所有需要调用的地方都共享这一单例对象。 单…