Java 集合面试题

1. 常见的集合有哪些?

Java集合类主要由两个根接口Collection和Map派生出来的,Collection派生出了三个子接口:List、Set、Queue(Java5新增的队列),因此Java集合大致也可分成List、Set、Queue、Map四种接口体系。

注意:Collection是一个接口,Collections是一个工具类,Map不是Collection的子接口。

Java集合框架图如下:

图中,List代表了有序可重复集合,可直接根据元素的索引来访问;Set代表无序不可重复集合,只能根据元素本身来访问;Queue是队列集合。

Map代表的是存储key-value对的集合,可根据元素的key来访问value。

上图中淡绿色背景覆盖的是集合体系中常用的实现类,分别是ArrayList、LinkedList、ArrayQueue、HashSet、TreeSet、HashMap、TreeMap等实现类。

2. 线程安全的集合有哪些?线程不安全的呢?

线程安全的:

  • Hashtable:比HashMap多了个线程安全。
  • ConcurrentHashMap:是一种高效但是线程安全的集合。
  • Vector:比Arraylist多了个同步化机制。
  • Stack:栈,也是线程安全的,继承于Vector。

线性不安全的:

  • HashMap
  • Arraylist
  • LinkedList
  • HashSet
  • TreeSet
  • TreeMap

3. Arraylist与 LinkedList 异同点?

  • 是否保证线程安全: ArrayList 和 LinkedList 都是不同步的,也就是不保证线程安全;
  • 底层数据结构: Arraylist 底层使用的是Object数组;LinkedList 底层使用的是双向循环链表数据结构;
  • 插入和删除是否受元素位置的影响: ArrayList 采用数组存储,所以插入和删除元素的时间复杂度受元素位置的影响。 比如:执行add(E e)方法的时候, ArrayList 会默认在将指定的元素追加到此列表的末尾,这种情况时间复杂度就是O(1)。但是如果要在指定位置 i 插入和删除元素的话(add(int index, E element))时间复杂度就为 O(n-i)。因为在进行上述操作的时候集合中第 i 和第 i 个元素之后的(n-i)个元素都要执行向后位/向前移一位的操作。 LinkedList 采用链表存储,所以插入,删除元素时间复杂度不受元素位置的影响,都是近似 O(1)而数组为近似 O(n)。
  • 是否支持快速随机访问: LinkedList 不支持高效的随机元素访问,而ArrayList 实现了RandomAccess 接口,所以有随机访问功能。快速随机访问就是通过元素的序号快速获取元素对象(对应于get(int index)方法)。
  • 内存空间占用: ArrayList的空 间浪费主要体现在在list列表的结尾会预留一定的容量空间,而LinkedList的空间花费则体现在它的每一个元素都需要消耗比ArrayList更多的空间(因为要存放直接后继和直接前驱以及数据)。

4. ArrayList 与 Vector 区别?

  • Vector是线程安全的,ArrayList不是线程安全的。其中,Vector在关键性的方法前面都加了synchronized关键字,来保证线程的安全性。如果有多个线程会访问到集合,那最好是使用 Vector,因为不需要我们自己再去考虑和编写线程安全的代码。
  • ArrayList在底层数组不够用时在原来的基础上扩展0.5倍,Vector是扩展1倍,这样ArrayList就有利于节约内存空间。

5. 说一说ArrayList 的扩容机制?

ArrayList扩容的本质就是计算出新的扩容数组的size后实例化,并将原有数组内容复制到新数组中去。默认情况下,新的容量会是原容量的1.5倍。

以JDK1.8为例说明:

public boolean add(E e) {
    //判断是否可以容纳e,若能,则直接添加在末尾;若不能,则进行扩容,然后再把e添加在末尾
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    //将e添加到数组末尾
    elementData[size++] = e;
    return true;
    }
// 每次在add()一个元素时,arraylist都需要对这个list的容量进行一个判断。通过ensureCapacityInternal()方法确保当前ArrayList维护的数组具有存储新元素的能力,经过处理之后将元素存储在数组elementData的尾部
private void ensureCapacityInternal(int minCapacity) {
      ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));
}
private static int calculateCapacity(Object[] elementData, int minCapacity) {
        //如果传入的是个空数组则最小容量取默认容量与minCapacity之间的最大值
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
            return Math.max(DEFAULT_CAPACITY, minCapacity);
        }
        return minCapacity;
    }

  private void ensureExplicitCapacity(int minCapacity) {
        modCount++;
        // 若ArrayList已有的存储能力满足最低存储要求,则返回add直接添加元素;如果最低要求的存储能力>ArrayList已有的存储能力,这就表示ArrayList的存储能力不足,因此需要调用 grow();方法进行扩容
        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }
private void grow(int minCapacity) {
        // 获取elementData数组的内存空间长度
        int oldCapacity = elementData.length;
        // 扩容至原来的1.5倍
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        //校验容量是否够
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        //若预设值大于默认的最大值,检查是否溢出
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        // 调用Arrays.copyOf方法将elementData数组指向新的内存空间
         //并将elementData的数据复制到新的内存空间
        elementData = Arrays.copyOf(elementData, newCapacity);
    }

6. Array 和 ArrayList 有什么区别?什么时候该应 Array 而不是 ArrayList 呢?

  • Array 可以包含基本类型和对象类型,ArrayList 只能包含对象类型。
  • Array 大小是固定的,ArrayList 的大小是动态变化的。
  • ArrayList 提供了更多的方法和特性,比如:addAll(),removeAll(),iterator() 等等。
全部评论

相关推荐

一、&nbsp;个人背景与综合意向-&nbsp;自我介绍:请进行简单的自我介绍。-&nbsp;业务背景:该规则引擎主要应用于唯品会的哪些具体业务场景?-&nbsp;团队情况:你所在组的整体开发团队规模是多少?纯后端开发人员有多少?-&nbsp;实习时间:如果入职,什么时候可以开始实习?打算实习多长时间?二、&nbsp;Java&nbsp;基础(重点考察&nbsp;HashMap)-&nbsp;数据结构:请介绍一下&nbsp;HashMap&nbsp;内部的数据结构。-&nbsp;设计原理:为什么&nbsp;HashMap&nbsp;要引入红黑树?这样设计的根本目的和好处是什么?-&nbsp;底层支撑:为什么底层要基于数组加链表的形式来实现哈希表?数组的作用是什么?-&nbsp;哈希计算:JDK&nbsp;1.8&nbsp;中,哈希函数是如何计算哈希值并确定元素在桶中位置的?(扰动函数的作用)-&nbsp;并发安全:HashMap&nbsp;是如何保证线程安全的?具体到&nbsp;JDK&nbsp;1.8,它是如何利用&nbsp;CAS&nbsp;和&nbsp;synchronized&nbsp;实现的?-&nbsp;不安全场景:在多线程环境下使用普通的&nbsp;HashMap&nbsp;会出现哪些具体的并发安全问题?(例如:值覆盖、扩容死循环等)三、&nbsp;Spring&nbsp;框架-&nbsp;核心机制:请解释一下&nbsp;Spring&nbsp;的循环依赖问题以及它是如何通过三级缓存解决的?-&nbsp;缓存细节:每一级缓存分别存的是什么?为什么非要引入“三级缓存”而不能只用二级缓存?(涉及&nbsp;Bean&nbsp;生命周期与&nbsp;AOP&nbsp;适配)四、&nbsp;项目深度挖掘(规则引擎)-&nbsp;项目核心:请详细介绍一下你负责的规则引擎项目,包括你的角色和核心产出。-&nbsp;难点与挑战:在该项目中,让你印象最深刻的技术难点或业务难点是什么?-&nbsp;稳定性治理:你是如何实现规则的多版本快照和回滚功能的?-&nbsp;数据库设计:为了兼容旧表并实现多版本功能,你在表结构上做了哪些具体修改?-&nbsp;业务逻辑:当规则出现问题时,业务人员如何通过后台系统排查过去的执行情况?-&nbsp;**回调机制**:业务回调功能是如何设计的?为什么要做异步优化?-&nbsp;可靠性考量:如果系统在回调过程中重启,线程池阻塞队列里的任务丢失了怎么办?如何保证数据一致性?-&nbsp;性能优化:你是如何发现规则执行链路存在耗时问题的?具体的优化手段(核心与非核心链路拆分)是什么?-&nbsp;系统指标:该系统的平均耗时(RT)和峰值&nbsp;QPS&nbsp;大约是多少?五、&nbsp;工程能力与运维排查-&nbsp;监控手段:你们是如何对线上规则的执行时间进行监控的?-&nbsp;线上排查:如果线上出现问题,你的排查流程是什么?-&nbsp;容器化知识:线上系统是容器化部署吗?你对&nbsp;Docker&nbsp;或&nbsp;K8S&nbsp;有多少了解?-&nbsp;日志定位:在容器环境下,你如何查找和定位历史日志文件?
查看25道真题和解析
点赞 评论 收藏
分享
评论
4
13
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务