如何自己实现一个简单的ArrayList

1.ArrayList有那些特性

  • 每个元素有自己索引值,元素可重复,元素可为null,

  • 基于数组实现,可自动进行扩容

  • 非线程安全

2.ArrayList有那些常用操作

最长见的操作有增删改查和遍历,因为ArrayList是基于数组实现的,所以会结合元素索引进行增删改查,这里就要注意检查传入值的合法性,如果操作一个不存在的索引就会引发数组越界,所以操作前必须进行索引检查。

  1. boolean add(E e)

添加一个元素到List尾部,一般在创建时会初始化一个默认长度的数组,jdk1.7中默认创建一个长度为0的数组,在add方法中进行扩容,默认数组长度是10,当添加第11个元素时数组会进行第二次扩容,每次扩大至原数组的1.5倍,扩容的实质就是创建一个更大的数组,再将原数组的内容复制到新数组中,原数组会被GC垃圾回收,数组最大容量为Integer.MAX_VALUE(2^31-1),超过最大容量,就会报数组越界异常,这个方法最重要的是扩容方法如何写。

  1. void add(int index, E element)

在指定的索引位置插入指定元素,这里传入索引值,先要判断索引是否越界,然后再判断是否需要扩容,在指定索引处插入元素,需要将该索引后面的元素全部后移一位,将指定值设置到指定索引,这个方法最重要的是中间插入元素,索引后的元素移位,最后组成一个新数组。从这里可以看出指定索引插元素的性能肯定比添加元素到末尾差,因为它每次都会创建新数组。

  1. E remove(int index)

首先检查索引是否越界,删除元素和上个方法刚好相反,它是将某个位置的元素删掉(其实是被后面元素覆盖),该位置后面的元素往前移一位,再将最后元素置null,容量减一

  1. boolean remove(Object o)

这个方法只传入了一个对象,所以需要遍历已有的元素,分别进行比较,找到匹配的元素的索引,后面的操作跟上个方法相同。这里需要注意,List中可能有null元素,o也可能是null,这样就不能全部用equals方法,先要判断o == null,分别用== 和equals方法比较。

  1. E set(int index, E element)

这个方法将指定索引的元素替换为新元素,实现比较简单,先检查索引是否越界,再替换指定位置的元素。

  1. E get(int index)

这个方法获取指定位置的元素,非常简单。

  1. int size()

这个方法返回List中当前元素个数,实现很简单,但是上面的增删操作都会去修改size值,注意这里的size是元素个数而不是数组长度。

  1. Iterator<E> iterator()

返回数组的迭代器,用于List遍历,实现比较复杂,下次单独写文章记录。

3.ArrayList实现原理

当前上面的操作都是一些常用的操作,实际ArrayList还有很多实用方法,这里只是简单实现下,了解其实现原理。实际工作中也不会用自己写的ArrayList。ArrayList实现原理中最重要的几个点有:

  • 内部用一个数组来存放元素

  • 底层使用system.arraycopy(Object src,int srcPos,Object dest,int destPos,int length)进行数组复制

  • 使用一个全局size记录元素个数

4.ArrayList实现代码

public class ArrayList<E> implements List<E> {
    
    private Object[] elementData;
    
    private int size;
    
    public ArrayList(int initCapcity){
        if(initCapcity < 0){
            throw new IllegalArgumentException("initCapcity 必须大于0");
        }
        elementData = new Object[initCapcity];
    }
    
    public ArrayList(){
        elementData = new Object[10];
    }

    @Override
    public void add(Object obj) {
        grow(size + 1);
        elementData[size++] = obj;
    }
    
    @Override
    public void add(int index, Object obj) {
        rangeCheckForAdd(index);
        grow(size + 1);
        System.arraycopy(elementData, index, elementData, index+1, size - index);
        elementData[index] = obj;
        size ++;
    }
    
    @Override
    public void remove(Object obj) {
        if(obj == null){
            for (int i = 0; i < size; i++) {
                if(elementData[i] == null){
                    fastRemove(i);
                }
            }
        }else{
            for (int i = 0; i < size; i++) {
                if(obj.equals(elementData[i])){
                    fastRemove(i);
                }
            }
        }
    }

    @Override
    public E remove(int index) {
        rangeCheck(index);
        int movedNum = size - index - 1;
        E oldElement = elementData(index);
        System.arraycopy(elementData, index+1, elementData, index, movedNum);
        elementData[--size] = null;
        return oldElement;
    }
    
    @Override
    public E get(int index) {
        rangeCheck(index);
        return elementData(index);
    }

    @Override
    public E set(int index, E obj) {
        rangeCheck(index);
        E oldElement = elementData(index);
        elementData[index] = obj;
        return oldElement;
    }

    @Override
    public int indexOf(E obj) {
        if(obj == null){
            for (int i = 0; i < size; i++) {
                if(elementData[i] == null){
                    return i;
                }
            }
        }else{
            for (int i = 0; i < size; i++) {
                if(obj.equals(elementData[i])){
                    return i;
                }
            }
        }
        return -1;
    }
    
    /**
     * 数组扩容
     * @param minCapacity
     */
    private void grow(int minCapacity) {
        if(minCapacity <= elementData.length){
            return;
        }
        int oldCapacity = elementData.length;
        int newCapacity = minCapacity + (oldCapacity >> 1);
        if(newCapacity < minCapacity){
            newCapacity = minCapacity;
        }
        if(minCapacity > Integer.MAX_VALUE){
            newCapacity = Integer.MAX_VALUE;
        }
        Object[] newArray = new Object[newCapacity];
        System.arraycopy(elementData, 0, newArray, 0, newCapacity);
        elementData = newArray;
    }
    
    @SuppressWarnings("unchecked")
     private E elementData(int index){
        return (E) elementData[index];
    }
    
    private void fastRemove(int i) {
        int numMoved = size - i -1;
        if(numMoved > 0){
            System.arraycopy(elementData, i+1, elementData, i, numMoved);
        }
        elementData[-- size] = null;
    }
    
    private void rangeCheck(int index){
        if(index >= size || index <0)
            throw new IndexOutOfBoundsException("index:"+index+",size:"+size);
    }
    
    private void rangeCheckForAdd(int index){
        if(index > size || index <0)
            throw new IndexOutOfBoundsException("index:"+index+",size:"+size);
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容