ArrayList的介绍和源码解析

ArrayList的介绍

1 ArrayList简介

ArrayList是List 接口的大小可变数组的实现。实现了所有可选列表操作,并允许包括 null 在内的所有元素。除了实现 List 接口外,此类还提供一些方法来操作内部用来存储列表的数组的大小。(此类大致上等同于 Vector 类,除了此类是不同步的。),此类是线程不安全的。

2 ArrayList的构造函数

public ArrayList()    //默认构造函数
public ArrayList(int initialCapacity)   //生成一个指定容量的空列表
public ArrayList(Collection<? extends E> c)  //生成一个包含指定元素的列表

3 ArrayList的Api

void        trimToSize    //把列表的容量修剪成列表的实际容量,删除动态增长的多余容量。
void        ensureCapacity(int minCapacity)   //根据传入的值对数组进行扩容,在数量已知的情况下
void        ensureCapacityInternal(int minCapacity)    //私有方法,根据传入的值对数组进行扩容
void        ensureExplicitCapacity(int minCapacity)   //传入数组进行扩容
void        grow(int minCapacity)     //进行扩容
int         hugeCapacity(int minCapacity)   //根据传入的值判断当前大小是否超过数组最大的大MAX_ARRAY_SIZE= Integer.MAX_VALUE - 8
int         size()    //获取列表大小
boolean     isEmpty()  //判断对象是否为空
boolean     contains(Object o)   //判断是否包含某个对象
int         indexOf(Object o)   //根据对象查询第一个索引下标
int         lastIndexOf(Object o)    //根据对象查询最后一个索引下标
Object      clone()   //对象进行复制
Object[]    toArray()  //将列表对象转成Object数组
T[]         toArray(T[] a)    //将列表对象转成某个固定对象的数组
E           elementData(int index)    //通过下标访问对象,内部方法
E           get(int index)    //通过下标访问对象
E           set(int index, E element)    //替换某个下标位置上的对象
boolean     add(E e)      //添加对象
void        add(int index, E element)   //在某个位置上添加对象,如果当前位置有相应的对象,则位置往右递增
E           remove(int index)     //删除某个对象通过下标
boolean     remove(Object o)   //通过对象查找删除某个对象 通过equals判断
void        fastRemove(int index)  //通过对象删除时的删除方法
void        clear()       //数据清空
boolean     addAll(Collection<? extends E> c)  //添加一个Collection队列到当前列表中
boolean     addAll(int index, Collection<? extends E> c)  //从某个位置开始添加一个Collection队列到当前列中
void        removeRange(int fromIndex, int toIndex)   //删除某个区间的数据
void        rangeCheck(int index)   //验证当前下标是否在实际数据范围内
void        rangeCheckForAdd(int index)   //添加数据到某个下标时 判断是否超过范围
String      outOfBoundsMsg(int index)  //数组越界显示字段
boolean     removeAll(Collection<?> c)    //删除传入的列表对象
boolean     retainAll(Collection<?> c)      //获取两个元素的交集 并把数据存入前面的列表中 l1.retainAll(l2)
boolean     batchRemove(Collection<?> c, boolean complement)   私有方法  //retainAll调用的内部方法    
void        writeObject(java.io.ObjectOutputStream s)    //
void        readObject(java.io.ObjectInputStream s)   //和WriteObject重写序列化方法
ListIterator<E> listIterator(int index)   //返回一个ListIterator从下标位置开始
ListIterator<E> listIterator()    //返回一个ListIterator接口的实现对象,下标从0开始
Iterator<E> iterator()  //返回Iterator接口具体实现对象
List<E>     subList(int fromIndex, int toIndex)  //返回开始和结束下标中的对象
void        subListRangeCheck(int fromIndex, int toIndex, int size)   //判断取区间对象是是否
void        forEach(Consumer<? super E> action)   //输出对象
Spliterator<E> spliterator()    //生成一个Spliterator对象,用于遍历和分割队列 一般配合Collection的stream方法使用,1.8新增方法
boolean     removeIf(Predicate<? super E> filter)   //移除满足条件的所有元素
void        sort(Comparator<? super E> c)   //列表进行排序

ArrayList的数据结构

ArrayList里面的重要字段说明

elementData:Object类型的数组,存放ArrayList添加进来的元素,是一个动态数组,大小会根据ArrayList的容量的增长而增长。
size:ArrayList中实际数据的大小。
modCount:实现fail-fast机制,不同线程对同一个集合进行操作是的错误机制

ArrayList的重要方法源码解析

//数据的添加
public boolean add(E e) {   //添加一个数据元素
        ensureCapacityInternal(size + 1);  // Increments modCount!!    //确定ArrayList的大小
        elementData[size++] = e;   //添加e元素到ArrayList中 并ArrayList大小加1
        return true;
    }

public void add(int index, E element) {  //添加一个元素到指定位置
        rangeCheckForAdd(index);   //判断当前位置是否越界(超出当前ArrayList大小和小于0)

        ensureCapacityInternal(size + 1);  // Increments modCount!!   //确定ArrayList的大小
        System.arraycopy(elementData, index, elementData, index + 1,
                         size - index);   //把新元素存放位置后面的元素往后移动一位
        elementData[index] = element;  //把新元素放到指定位置
        size++;  //ArrayList大小加1
    }

public boolean addAll(Collection<? extends E> c) {  //往ArrayList中添加一个Collection集合
        Object[] a = c.toArray();
        int numNew = a.length;
        ensureCapacityInternal(size + numNew);  // Increments modCount  //把传入集合转为数组对象,通过长度确定ArrayList大小
        System.arraycopy(a, 0, elementData, size, numNew);  //把转为数组对象的Collection集合添加到ArrayList中
        size += numNew;
        return numNew != 0;
    }

public boolean addAll(int index, Collection<? extends E> c) {  //添加一个Collection集合到ArrayList中的指定位置
        rangeCheckForAdd(index);

        Object[] a = c.toArray();
        int numNew = a.length;
        ensureCapacityInternal(size + numNew);  // Increments modCount  把传入集合转为数组对象,通过长度确定ArrayList大小

        int numMoved = size - index;
        if (numMoved > 0)
            System.arraycopy(elementData, index, elementData, index + numNew,
                             numMoved);    //计算指定位置插入数据是原数据往后移动位置,并移动

        System.arraycopy(a, 0, elementData, index, numNew);  //把转为数组对象的Collection集合添加到ArrayList中
        size += numNew; //确定size
        return numNew != 0;
    }

private void ensureCapacityInternal(int minCapacity) {    //确定ArrayList的大小
        if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {   //判断是否为默认创建的空元素
            minCapacity = Math.max(DEFAULT_CAPACITY, minCapacity);  //大小为默认大小和传进来大小去最大值
        }

        ensureExplicitCapacity(minCapacity);  
    }

private void ensureExplicitCapacity(int minCapacity) {  //判断并确定ArrayList大小
        modCount++;   //证明进行了ArrayList的数据添加操作

        // overflow-conscious code
        if (minCapacity - elementData.length > 0)   //如果传进来的大小大于现在数组大小,就对数组进行自增操作
            grow(minCapacity);
    }

private void grow(int minCapacity) {
        // overflow-conscious code
        int oldCapacity = elementData.length;
        int newCapacity = oldCapacity + (oldCapacity >> 1);  //自增的大小默认为原大小的1.5倍
        if (newCapacity - minCapacity < 0)   //如果自增大小小于添加数据后的带下,新的自增大小更改为传入数据大小
            newCapacity = minCapacity;
        if (newCapacity - MAX_ARRAY_SIZE > 0)   //如果自增大小大于ArrayList中自定义最大大小,就把大小设置为int值得最大大小范围
            newCapacity = hugeCapacity(minCapacity);
        // minCapacity is usually close to size, so this is a win:
        elementData = Arrays.copyOf(elementData, newCapacity);  //数组进行自增
    }

private static int hugeCapacity(int minCapacity) {
        if (minCapacity < 0) // overflow
            throw new OutOfMemoryError();
        return (minCapacity > MAX_ARRAY_SIZE) ?
            Integer.MAX_VALUE :
            MAX_ARRAY_SIZE;
    }
//ArrayList数据删除
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;
    }

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
    }

protected void removeRange(int fromIndex, int toIndex) {  //删除一个范围内的数据
        modCount++;
        int numMoved = size - toIndex;  //通过删除最后的下标往前移动 然后把取消的数据设为null值
        System.arraycopy(elementData, toIndex, elementData, fromIndex,
                         numMoved);

        // clear to let GC do its work
        int newSize = size - (toIndex-fromIndex);
        for (int i = newSize; i < size; i++) {
            elementData[i] = null;
        }
        size = newSize;
    }

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; // clear to let GC do its work  //最后一个元素设为null值

        return oldValue;
    }

public boolean removeAll(Collection<?> c) {  //删除ArrayList中存在的输入的集合的元素 
        Objects.requireNonNull(c);  //判断是否为空
        return batchRemove(c, false);
    }

private boolean batchRemove(Collection<?> c, boolean complement) {
        final Object[] elementData = this.elementData;
        int r = 0, w = 0;
        boolean modified = false;  
        try {
            for (; r < size; r++)
                if (c.contains(elementData[r]) == complement)   //complement:true 返回相同的元素 false 返回不同元素
                    elementData[w++] = elementData[r];
        } finally {
            if (r != size) {  //如果不等于证明前面报了异常,把后面的部分补全
                System.arraycopy(elementData, r,
                                 elementData, w,
                                 size - r);
                w += size - r;
            }
            if (w != size) {    //如果w!=size说明进行了删除操作,故需将删除的值赋为null
                for (int i = w; i < size; i++)
                    elementData[i] = null;
                modCount += size - w;
                size = w;
                modified = true;
            }
        }
        return modified;
    }
public E get(int index) {  //通过下标返回数据 先判断下标是否正确
        rangeCheck(index);

        return elementData(index);
    }

ArrayList的遍历方式

1、通过迭代器的方式去实现,既Iterator的方式

Iterator iterator = list.iterator;
while(iterator.hasNext()){
    String value = iterator.next();
}

2、通过索引随机访问,ArrayList默认实现了RandomAccess

  for(int i = 0;i<size;i++){
    String value = list.get(i);
}

3、通过ForEach循环遍历

 for(String str:list){
    String value = str;
}

通过对比三种方式中随机访问的效率,迭代器的方式效率最低

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容