List的两种实现对比

前言

在日常的java开发中,我们经常用到各种集合类,而List是其中最常见的一种;以前我们在使用数组的时候,无论是c++或者java,都要指定它的大小;而List居然不用指定大小,太神奇了,接下来让我们仔细分析一下它们的实现

LinkedList:

  1. 主要成员变量:
transient int size = 0; //List长度
transient Node<E> first;//头指针
transient Node<E> last;//尾指针
  1. add方法(将元素放入List尾部):
public boolean add(E e) {
    linkLast(e);
    return true;
}

再看linkLast方法:

void linkLast(E e) {
    final Node<E> l = last;
    final Node<E> newNode = new Node<>(l, e, null);
    last = newNode;
    if (l == null)
        first = newNode;
    else
        l.next = newNode;
    size++;
    modCount++;
}

也就是说,每次添加一个节点到末尾时,都是新申请一个链表节点,然后再缝缝补补,把相关的链表指针接在一起;复杂度为O(1);

再看看另外一个add方法:

  1. add(int index, E element)(将元素插入到某个下标位置):
public void add(int index, E element) {
    checkPositionIndex(index);

    if (index == size)
        linkLast(element);
    else
        linkBefore(element, node(index));
}

<p>
再看node()方法(找到下标对应的链表节点):

Node<E> node(int index) {
    // assert isElementIndex(index);

    if (index < (size >> 1)) {
        Node<E> x = first;
        for (int i = 0; i < index; i++)
            x = x.next;
        return x;
    } else {
        Node<E> x = last;
        for (int i = size - 1; i > index; i--)
            x = x.prev;
        return x;
    }
}

其寻址复杂度是O(n)的,唯一的优化点就在于,看下标离头指针近还是离尾指针近,复杂度做了常数上的优化,从n下降到n/2, 但还是O(n);
剩下的删查都涉及到寻址的时间损耗,复杂度也都是O(n),它们的实现都是类似的,就不赘诉了。

小优化:其实如果我们自己实现一个LinkedList,完全可以把指针信息加在节点上,这样就可以避免一些寻址上的时间损耗了。

ArrayList(与c++里的vector类似)

  1. 成员变量:
transient Object[] elementData; // 存储元素的载体是数组
private int size;//List当前长度
  1. 主构造方法:
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);
    }
}
没有什么特别的,就是申请一个定长的数组
  1. add(E e)方法:
public boolean add(E e) {
    ensureCapacityInternal(size + 1);  // Increments modCount!!
    elementData[size++] = e;
    return true;
}

第一行是确保数组载体容量足够,后边就是把下标为size+1的元素赋值成e;
它调用了ensureCapacityInternal()方法:

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

    ensureExplicitCapacity(minCapacity);
}

又调用了ensureExplicitCapacity()方法:

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

    // overflow-conscious code
    if (minCapacity - elementData.length > 0)
        grow(minCapacity);
}

再看grow()方法:

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);
}

里边关键的一句代码就是:

int newCapacity = oldCapacity + (oldCapacity >> 1);

意思就是,新的数组容量变成原数组的大约1.5倍(看不懂的可以去了解一下位运算),新的数组申请内存之后,再把老的数组里的元素全copy到新数组就完成了;
那么为什么每次容量不足的时候,要扩充到1.5倍呢?通过分析我们可以发现,每次扩充1.5倍容量,那么假设最后容量扩充为n,那么总体上的申请空间的数量近似于:f(n) = n + n * 2/3 + n * (2/3)^2 + n * (2/3)^3 + ... ,根据等比数列公式可以知道,这个表达式的值为: f(n) = 3 * n = O(n)
采用倍增的方法扩容,优点在于总体的复杂度数量级是线性的,但是也不可避免的可能会有空间浪费;在最极端的情况下,会有接近1/3的空间是没有利用上的;因此,在List会很大,且能预先大致估算出List会有多大的前提下,为了减少系统的内存消耗和频繁GC,应尽量使用以下构造方法来申请List:

public ArrayList(int initialCapacity);
  1. add(int index, E element)方法(将元素插入到指定下标):
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++;
}

跟add(E element)方法的最大的差别就在于,在赋值之前,要先把该下标以及后面的元素全都向后移一位,其复杂度为O(n);

  1. 接下来看get(int index)方法:
public E get(int index) {
    rangeCheck(index);

    return elementData(index);
}

检查下标合法性,再调用elementData():

E elementData(int index) {
    return (E) elementData[index];
}

直接是数组的根据下标随机访问的操作,复杂度是O(1),再强转成E类型就返回了;

  1. remove(int index)方法:
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

    return oldValue;
}

在删除指定下标的元素时,如果这个元素不是最后一个,那么要将后面的元素全部向前移一位,复杂度也是O(n), 另外还有一点就是,删除了这个元素之后,要把最大下标的那个元素赋值成null,方便系统进行GC(系统GC时,会根据对象是否被引用来判断对象是否可以回收)

  1. remove(Object o)方法:
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;
}

定位到元素位置复杂度为O(n), fastRemove(index)的复杂度也是O(n);
另外,初看这段代码,感觉写得有点“啰嗦”,但实际上确实是有必要的:List元素允许为空,所以要特判该元素是否为空,为空时直接用==null来判断,而不为空时,才能调用equals方法进行比较,否则会有空指针异常;所以这段代码也是JDK严谨性的一种体现;

  1. 以下表格列出了两者在各种操作下的复杂度:

| 操作\List实现类 | LinkedList | ArrayList |
| ------------- |:-------------:| -----:|
| add(E e) | O(1) | O(1) |
| add(int index, E element) | O(n) | O(n) |
| get(int index) | O(n) | O(1) |
| remove(int index) | O(n) | O(n) |
| remove(Object o) | O(n) | O(n) |

从上表看,LinkedList几乎没有比ArrayList优越的地方;另外,LinkedList相比ArrayList没有0.5倍的空间浪费,但是其每个节点都有前后指针的内存占用,且每次新增一个元素时都要新申请一个Node,而ArrayList则是一次性批量申请;所以,当List长度比较大的时候,肯定是ArrayList效率比较高。
如果说LinkedList有优点的话,可能就是它不需要申请连续的内存,所以建议大家除了极端情况,大部分时候都使用ArrayList。

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

推荐阅读更多精彩内容