Android面试题之ArrayList源码详解

简介: ArrayList是Java中基于数组实现的列表,提供O(1)的索引访问,但插入和删除操作平均时间复杂度为O(n)。默认容量为10,当需要时会通过System.arraycopy扩容。允许存储null,非线程安全。面试常问:List是接口,ArrayList是其实现之一,推荐使用List接口编程以实现更好的灵活性。更多详情见[ArrayList源码](http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/8u40-b25/java/util/ArrayList.java#ArrayList.Node)。

本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”关注,和我一起每天进步一点点

ArrayList

public class ArrayList<E> extends AbstractList<E>
            implements List<E>, RandomAccess, Cloneable, java.io.Serializable
{
   
    /**
     * 默认ArrayList的容量为10
     */
    private static final int DEFAULT_CAPACITY = 10;

    /**
     * 可以看出ArrayList底层通过数组实现
     */
    private static final Object[] EMPTY_ELEMENTDATA = {
   };

    private static final Object[] DEFAULTCAPACITY_EMPTY_ELEMENTDATA = {
   };


    /**
     * 可以看出ArrayList基于数组实现
     */
    transient Object[] elementData;

    private int size;

    /**
     * 
     */
    public ArrayList() {
   
        this.elementData = DEFAULTCAPACITY_EMPTY_ELEMENTDATA;
    }

    public ArrayList(int initialCapacity) {
   
        if (initialCapacity > 0) {
   
            this.elementData = new Object[initialCapacity];
        } else if (initialCapacity == 0) {
   
            this.elementData = EMPTY_ELEMENTDATA;
        } else {
   
            throw new IllegalArgumentException("Illegal Capacity: "+
                                               initialCapacity);
        }
    }


     /**
     *获取大小
     */
    public int size() {
   
        return size;
    }

    /**
     *判断是否为空
     */
    public boolean isEmpty() {
   
        return size == 0;
    }

    public boolean contains(Object o) {
   
        return indexOf(o) >= 0;
    }

    /**
     * 找出目标的下标,不存在返回-1
     * 可以看出ArrayList中可以存储null
     */
    public int indexOf(Object o) {
   
        if (o == null) {
   
            for (int i = 0; i < size; i++)
                if (elementData[i]==null)
                    return i;
        } else {
   
            for (int i = 0; i < size; i++)
                if (o.equals(elementData[i]))
                    return i;
        }
        return -1;
    }

    /**
     * indexOf方法从前往后找,lastIndexOf从后往前找
     * 
     */
    public int lastIndexOf(Object o) {
   
        if (o == null) {
   
            for (int i = size-1; i >= 0; i--)
                if (elementData[i]==null)
                    return i;
        } else {
   
            for (int i = size-1; i >= 0; i--)
                if (o.equals(elementData[i]))
                    return i;
        }
        return -1;
    }

    /**
     * 通过Arrays.copyOf方法返回包含所有数据的数组
     */
    public Object[] toArray() {
   
        return Arrays.copyOf(elementData, size);
    }

    /**
     * 检查数组是否越界,这里就是抛出IndexOutOfBoundsException的地方
     */
    private void rangeCheck(int index) {
   
        if (index >= size)
            throw new IndexOutOfBoundsException(outOfBoundsMsg(index));
    }

    /**
     * 获取元素很方便
     */
    public E get(int index) {
   
        //检查下标是否越界
        rangeCheck(index);
        return (E) elementData[index];
    }

    /**
     * 更新指定位置的元素
     */
    public E set(int index, E element) {
   
        rangeCheck(index);

        E oldValue = elementData(index);
        elementData[index] = element;
        return oldValue;
    }

    /**
     * 添加元素,每次添加之前都会检查下容量够不够,不够的话就会进行扩容
     */
    public boolean add(E e) {
   
        ensureCapacityInternal(size + 1);  // Increments modCount!!
        elementData[size++] = e;
        return true;
    }

    /**
     * 添加元素到指定位置,这个就需要移动元素了,而移动元素是通过System.arraycopy方法来拷贝数组,相对复杂度比较高
     */
    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++;
    }

    private void ensureCapacityInternal(int minCapacity) {
   
        if (elementData == EMPTY_ELEMENTDATA) {
   
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);
        }

        ensureExplicitCapacity(minCapacity);
    }

    private void ensureExplicitCapacity(int minCapacity) {
   
        modCount++;

        if (minCapacity - elementData.length > 0)
            grow(minCapacity);
    }

    /**
     * grow方法进行扩容,最后调用的是Arrays.copyOf方法将老数据拷贝到新数组里面
     */
    private void grow(int minCapacity) {
   
        // overflow-conscious code
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);
        if (newCapacity - minCapacity < 0)
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)
            newCapacity = hugeCapacity(minCapacity);
        // minCapacity is usually close to size, so this is a win:
        elementData = Arrays.copyOf(elementData, newCapacity);
    }

    /**
     * 删除元素也和添加元素一样分为删除指定位置元素和删除指定元素的情况
     * 不论哪种情况数组的删除复杂度都高
     */
    public E remove(int index) {
   
        rangeCheck(index);

        modCount++;
        E oldValue = elementData(index);

        int numMoved = size - index - 1;
        // 如果删除的是中间位置的元素就需要移动数组了,否则就不用移动数组
        if (numMoved > 0)
            System.arraycopy(elementData, index+1, elementData, index,
                             numMoved);
        // 释放空出来的位置                     
        elementData[--size] = null; 

        return oldValue;
    }

    /**
     * 删除元素就没有获取元素那么方便了,得先遍历找到要删除的元素
     * 这里可以看到,ArrayList是允许值是null的
     */
    public boolean remove(Object o) {
   
        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;
    }

    /*
     * 调用System.arraycopy方法实现删除元素
     * 并且这里过滤了删除的是末尾元素的情况
     */
    private void fastRemove(int index) {
   
        modCount++;
        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 work
    }

    /*
     * 清空数组就是把数组的各个位置置null
     */
    public void clear() {
   
        modCount++;

        // clear to let GC do its work
        for (int i = 0; i < size; i++)
            elementData[i] = null;

        size = 0;
    }

}

总结

1、 底部基于数组实现,这样的话查找比较快,复杂度为O(1),但是插入和删除数据就比较慢了,而且数据量越大插入和删除的速度越慢。复杂度为O(n)

2、 默认容量大小为10,超过这个容量就会进行扩容,扩容的话最终调用的是System.arraycopy方法,这是一个native方法

3、 ArrayList里面允许存储null值

4、 ArrayList不是线程安全的,只能用于单线程环境下

面试常问的一个问题

关于ArrayList一个常问的问题就是List和ArrayList的区别

实际上答案很简单,就是一个是接口一个是具体实现的关系,ArrayList是List的其中一个具体实现类。然后我们在实际开发中一般是这么应用:

List<String> list = new ArrayList<String>();

其实这是一种面向接口的思路,我们在需要list的地方引用的都是List接口类型,这样的话如果后面我们需要将ArrayList改为LinkedList的话只需要将上面代码改为:

List<String> list = new LinkedList<String>();

其他引用到list的地方都不需要改,这就是依赖接口的好处。

以上是基于Java1.8并且只介绍了常用的一些方法的原理,详细的ArrayList源码请查看:ArrayList源码

欢迎关注我的公众号AntDream查看更多精彩文章!

目录
相关文章
|
4天前
|
Java Android开发
Android面试题经典之Glide取消加载以及线程池优化
Glide通过生命周期管理在`onStop`时暂停请求,`onDestroy`时取消请求,减少资源浪费。在`EngineJob`和`DecodeJob`中使用`cancel`方法标记任务并中断数据获取。当网络请求被取消时,`HttpUrlFetcher`的`cancel`方法设置标志,之后的数据获取会返回`null`,中断加载流程。Glide还使用定制的线程池,如AnimationExecutor、diskCacheExecutor、sourceExecutor和newUnlimitedSourceExecutor,其中某些禁止网络访问,并根据CPU核心数动态调整线程数。
14 2
|
3天前
|
Android开发
Android面试题经典之如何全局替换App的字体
在Android应用中替换字体有全局和局部方法。全局替换涉及在`Application`的`onCreate`中设置自定义字体,并创建新主题。局部替换则可在布局中通过`ResourcesCompat.getFont()`加载字体文件并应用于`TextView`。
15 2
|
4天前
|
算法 Java API
Android性能优化面试题经典之ANR的分析和优化
Android ANR发生于应用无法在限定时间内响应用户输入或完成操作。主要条件包括:输入超时(5秒)、广播超时(前台10秒/后台60秒)、服务超时及ContentProvider超时。常见原因有网络、数据库、文件操作、计算任务、UI渲染、锁等待、ContentProvider和BroadcastReceiver的不当使用。分析ANR可借助logcat和traces.txt。主线程执行生命周期回调、Service、BroadcastReceiver等,避免主线程耗时操作
18 3
|
9天前
|
SQL XML Java
Android 这 13 道 ContentProvider 面试题,你都会了吗?
Android 这 13 道 ContentProvider 面试题,你都会了吗?
|
9天前
|
安全 Android开发 Kotlin
Android面试题之Kotlin协程并发问题和互斥锁
Kotlin的协程提供轻量级并发解决方案,如`kotlinx.coroutines`库。`Mutex`用于同步,确保单个协程访问共享资源。示例展示了`withLock()`、`lock()`、`unlock()`和`tryLock()`的用法,这些方法帮助在协程中实现线程安全,防止数据竞争。
13 1
|
2天前
|
JavaScript Java 测试技术
基于springboot+vue.js+uniapp的高校后勤网上报修系统安卓app附带文章源码部署视频讲解等
基于springboot+vue.js+uniapp的高校后勤网上报修系统安卓app附带文章源码部署视频讲解等
5 0
|
6天前
|
缓存 编解码 安全
Android经典面试题之Glide的缓存大揭秘
Glide缓存机制包括内存和硬盘缓存。内存缓存使用弱引用的ActiveResources和LRU策略,硬盘缓存利用DiskLruCache。Engine.load方法首先尝试从内存和弱引用池加载,然后从LRU缓存中加载图片,增加引用计数并移出LRU。若缓存未命中,启动新任务或加入现有任务。内存大小根据设备内存动态计算,限制在0.4以下。DiskLruCache使用自定义读写锁,保证并发安全,写操作通过锁池管理,确保高效。
8 0
|
10天前
|
存储 安全 Java
《ArrayList & HashMap 源码类基础面试题》面试官们最喜欢问的ArrayList & HashMap源码类初级问,你都会了?
《ArrayList & HashMap 源码类基础面试题》面试官们最喜欢问的ArrayList & HashMap源码类初级问,你都会了?
11 0
|
10天前
|
Java API Android开发
技术经验分享:Android源码笔记——Camera系统架构
技术经验分享:Android源码笔记——Camera系统架构
11 0
|
存储 缓存 Java
Android--面试中遇到的问题总结(一)
版权声明:本文为博主原创文章,转载请标明出处。 https://blog.csdn.net/chaoyu168/article/details/56479430 一、handl...
1309 0