1,ArrayList面试必问

  • 说说ArrayList和LinkedList的区别?

    ArrayList基于数组实现,LinkedList基于链表实现,不同的数据结构决定了ArrayList查询效率比较高,而LinkedList插入删除效率比较高,反过来就比较慢了。

  • ArrayList默认初始容量为多少?按照几倍来扩容?

    10,1.5倍。

  • 说说数组扩容的原理?

    ArrayList扩容调用的是Array.copyof函数,把老数组遍历赋值给新数组返回。

  • 说说ArrayList常见方法的时间复杂度?

    • get方法通过下标获取元素,时间复杂度为O(1)
    • add方法直接添加会添加到集合的尾部,时间复杂度为O(1)
    • add方法通过下标添加到非尾部会引起数组的批量移动,时间复杂度为O(n),否则为O(1)
    • remove方法通过下标删除非尾部元素引起数组批量移动,时间复杂度为O(n),反之则为O(1)
    • remove方法根据对象删除需要遍历集合,时间复杂度为O(n),如果删除的为非尾部元素,会使数组批量移动,时间复杂度为O(n^2)

    总之,通过下标操作的时间复杂度为O(1),如果触发了数组的批量移动,时间复杂度为O(n),如果通过对象操作需要遍历集合,时间复杂度已经为O(n),若同时触发了数组的移动,时间复杂度为O(n^2).

  • ArrayList和vector的区别

    • 最大的区别在于线程是否安全
    • 其次Vector是两倍扩容
    • 最后就是在不指定大小的情况下,ArrayList容量初始化是在添加元素的时候,而Vector有一个无参构造器直接初始化为10

2,Debug ArrayList源码

由于1.7和1.8几乎没什么变化,本文以jdk1.8为例。

2.1 用Debug分析一个元素是如何add进ArrayList

编写测试用例,打上断点:

先分析构造函数如何初始化,关键步骤如下:

elementData是ArraList底层数组的实现,(ps:hashMap数组使用table命名)

DEFAULTCAPACITY_EMPTY_ELEMENTDATA表示默认的空数组,也就是说ArrayList在构造函数初始化时并不会进行底层数组的初始化。

给元素的添加打上断点,分析过程:

进入add方法内部:

public boolean add(E e) {
//确保内部容量,在元素添加进来前可能要进行扩容操作,size初始化为0,表示集合的长度
ensureCapacityInternal(size + 1); // Increments modCount!!
//添加元素,size自增
elementData[size++] = e;
return true;
}

进入ensureCapacityInternal方法内部:此时elementData为空,size+1=minCapacity=1

ensureExplicitCapacity:确保明确的能力

计算容量,calculateCapacity方法:

private static int calculateCapacity(Object[] elementData, int minCapacity) {
//判断数组是否为空,若为空,返回默认容量和最小容量的最大值,若不为空,返回最小容量
if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {
return Math.max(DEFAULT_CAPACITY, minCapacity);
}
return minCapacity;
}

DEFAULT_CAPACITY默认容量为10:

继续分析,进入:

ensureExplicitCapacity(calculateCapacity(elementData, minCapacity));此时参数为10,也就是ArrayList的默认容量
private void ensureExplicitCapacity(int minCapacity) {
modCount++; //集合的修改次数 //如果最小容量减去数组长度大于0,进行扩容,此时最小容量为10,数组长度为0
if (minCapacity - elementData.length > 0)
grow(minCapacity);
}

核心扩容函数grow:(ps:HashMap中扩容函数为resize)

private void grow(int minCapacity) {
//oldCapacity:旧数组容量
int oldCapacity = elementData.length; //新容量等于旧容量加上旧容量的一半,>>1相当于除以2(ArrayList扩容是1.5倍)
int newCapacity = oldCapacity + (oldCapacity >> 1); //新容量小于最小容量,则赋值为最小容量,此时newCapacity等于0,minCapacity为10
if (newCapacity - minCapacity < 0)
newCapacity = minCapacity; //MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8
if (newCapacity - MAX_ARRAY_SIZE > 0)
newCapacity = hugeCapacity(minCapacity);
//赋值给新数组
elementData = Arrays.copyOf(elementData, newCapacity);
}

数组复制Arrays.copyOf:

2.2 用Debug分析如何通过数组下标获取ArrayList元素

打上断点,debug:

首先进行范围检查,而后返回元素

2.3 用Debug分析如何通过数组下标删除一个元素

打上断点:

进入remove方法内部,

 public E remove(int index) {
//下标范围检查
rangeCheck(index);
//修改次数自增
modCount++;
//保留当前删除元素的值,稍后返回
E oldValue = elementData(index);
//需要移动元素的个数
int numMoved = size - index - 1;
if (numMoved > 0)
//底层使用native方法,debug进不去。native方法:java调用其他语言的接口
System.arraycopy(elementData, index+1, elementData, index,
numMoved);
//最后一位置空
elementData[--size] = null; // clear to let GC do its work
//返回删除元素的值
return oldValue;
}

2.4 用Debug分析如何通过对象删除一个元素

进入remove方法:

public boolean remove(Object o) {
//如果对象为空,则遍历ArrayList集合
if (o == null) {
for (int index = 0; index < size; index++)
if (elementData[index] == null) {
fastRemove(index);
return true;
}
} else {
//不为空,也遍历
for (int index = 0; index < size; index++)
if (o.equals(elementData[index])) {
fastRemove(index);
return true;
}
}
return false;
}

进入fastRemove方法:

 private void fastRemove(int index) {
modCount++; //numMoved:需要移动数组的个数
int numMoved = size - index - 1;
if (numMoved > 0)
System.arraycopy(elementData, index+1, elementData, index,
numMoved);
elementData[--size] = null; // clear to let GC do its wrk
}

2.5 用Debug分析向数组中间添加元素

进入add方法

public void add(int index, E element) {
//下标范围检查
rangeCheckForAdd(index);
//确保容量大小
ensureCapacityInternal(size + 1); // Increments modCount!! //移动数组位置
System.arraycopy(elementData, index, elementData, index + 1,
size - index);
//赋值给数组
elementData[index] = element;
//元素个数增加
size++;
}

关于System.arraycopy时间复杂度问题,在添加或者删除最后一个元素的时候不会触发数组的复制机制,时间复杂度为O(1),若是添加到数组中间,由于会触发数组的复制,时间复杂度为O(n)。对于删除元素同样,根据数组下标删除的情况下,删除尾部元素是不会触发数组的扩容机制的,若删除中间的元素,同样会触发数组的复制。若根据对象删除元素,由于本身遍历到对象的时间复杂度为O(n),删除元素后再对数组进行重组,所以时间复杂度为O(n^2)。

DEBUG ArrayList的更多相关文章

  1. MAT使用--转

    原文地址: [1]http://ju.outofmemory.cn/entry/172684 [2]http://ju.outofmemory.cn/entry/129445 MAT使用入门 MAT简 ...

  2. 韩顺平Java(持续更新中)

    原创上课笔记,转载请注明出处 第一章 面向对象编程(中级部分) PDF为主 1.1 IDEA 删除当前行,ctrl+y 复制当前行,ctrl+d 补全代码,alt+/ 添加或者取消注释,ctrl+/ ...

  3. jmeter sampler maven项目排错记

    eclipse 创建的maven项目,引入jar包之后出现红色叹号,一直找不到原因,连main方法都无法运行,提示找不到类: 错误: 找不到或无法加载主类 soapsampler.SoapSample ...

  4. JAVA基础学习之IP简述使用、反射、正则表达式操作、网络爬虫、可变参数、了解和入门注解的应用、使用Eclipse的Debug功能(7)

    1.IP简述使用//获取本地主机ip地址对象.InetAddress ip = InetAddress.getLocalHost();//获取其他主机的ip地址对象.ip = InetAddress. ...

  5. Java中ArrayList相关的5道面试题

    本文参考了 <关于ArrayList的5道面试题 > 1.ArrayList的大小是如何自动增加的? 这个问题我想曾经debug过并且查看过arraylist源码的人都有印象,它的过程是: ...

  6. C#之ArrayList

    using System.Collections; 新建: ArrayList list = new ArrayList(); 添加元素: int a = 1; list.Add(a); 遍历: fo ...

  7. Handler 接收Parcelable ArrayList时返回空的错误

    遇到一个问题,从handler 接收的Parcelable ArrayList返回空,调试发现这个arraylist生成的时候是有值的,传到handler就没值了 赋值的代码 new Thread(n ...

  8. 【转】Java如何克隆集合——深度拷贝ArrayList和HashSet

    原文网址:http://blog.csdn.net/cool_sti/article/details/21658521 原英文链接:http://javarevisited.blogspot.hk/2 ...

  9. Java入门(7)——循环和debug 调试

    循环: while 循环: 格式: int i = 0; ①    //初始化条件 while(i < 10) { ②  //判断条件 System.out.println(i); ④ //循环 ...

随机推荐

  1. Flutter学习笔记(35)--通知Notification

    如需转载,请注明出处:Flutter学习笔记(35)--通知Notification 通知的NotificationListener和我们之前写的事件的Listener一样,都是功能性的组件,而且也都 ...

  2. Redis安装过程jemalloc/jemalloc.h报错

    问题: [root@localhost redis-3.0.0]# make cd src && make all make[1]: Entering directory `/data ...

  3. NodeJs通过HTTP模块发起GET|POST请求

    [本文版权归微信公众号"代码艺术"(ID:onblog)所有,若是转载请务必保留本段原创声明,违者必究.若是文章有不足之处,欢迎关注微信公众号私信与我进行交流!] Node.js ...

  4. 人脸识别Demo解析C#

    概述 不管你注意到没有,人脸识别已经走进了生活的角角落落,钉钉已经支持人脸打卡,火车站实名认证已经增加了人脸自助验证通道,更别提各个城市建设的『智能城市』和智慧大脑了.在人脸识别业界,通常由人脸识别提 ...

  5. MQ消息队列(1)—— 概念和使用场景

    一.什么是消息队列  消息即是信息的载体.为了让消息发送者和消息接收者都能够明白消息所承载的信息(消息发送者需要知道如何构造消息:消息接收者需要知道如何解析消息),它们就需要按照一种统一的格式描述消息 ...

  6. skywalking中文文档

    https://github.com/apache/skywalking/blob/v5.0.0-alpha/docs/README_ZH.md 大家可以前往如下地址下载我们的发布包: l  Apac ...

  7. 动力节点 mysql 郭鑫 34道经典的面试题三

    1.第十五题 15.列出受雇日期早于其直接上级的所有员工编号.姓名.部门名称 思路一:第一步将emp a看成员工表,将emp b 看成领导表,员工表的mgr字段应该等于领导表的主键字段 mysql&g ...

  8. WPF中的Data Binding调试指南

    大家平时做WPF开发,相信用Visual studio的小伙伴比较多.XAML里面曾经在某些特殊版本的Visual Studio中是可以加断点进行调试的,不过目前多数版本都不支持在XAML加断点来调试 ...

  9. java中的Arrays这个工具类你真的会用吗

    Java源码系列三-工具类Arrays ​ 今天分享java的源码的第三弹,Arrays这个工具类的源码.因为近期在复习数据结构,了解到Arrays里面的排序算法和二分查找等的实现,收益匪浅,决定研读 ...

  10. 服务认证(JWT)

    上一篇已经讲了微服务组件中的 路由网关(Zuul),但是未介绍服务认证相关,本章主要讲解基于Spring Security 与 JJWT 实现 JWT(JSON Web Token)为接口做授权处理… ...