Java容器相关(1)-- List

其中TreeMap、TreeSet有序指的是里面的元素按照大小排序,List、LinkedHashMap、LinkedHashSet有序指的是元素按照插入的顺序存储。


    队列继承图,Deque接口又有ArrayDeque和LinkedList、ConcurrentLinkedDeque实现类及BlockingDeque接口,其中BlockingDeque接口有LinkedBlockingDeque并发实现类。

一、List

 1)ArrayList:

    概括的说,ArrayList是一个动态数组,它是线程不安全的,允许元素为null。

其底层数据结构依然是数组,它实现了List<E>, RandomAccess, Cloneable,

java.io.Serializable接口,其中RandomAccess代表了其拥有随机快速访问的能力,ArrayList可以以O(1)的时间复杂度去根据下标访问元素。

    因其底层数据结构是数组,所以可想而知,它是占据一块连续的内存空间(容量就是数组的length,数组的默认大小为10),所以它也有数组的缺点,空间效率不高。由于数组的内存连续,可以根据下标以O1的时间读写(改查)元素,因此时间效率很高。

    当集合中的元素超出这个容量,便会进行扩容操作。扩容操作也是ArrayList的一个性能消耗比较大的地方,所以若我们可以提前预知数据的规模,应该通过public ArrayList(int initialCapacity) {}构造方法,指定集合的大小,去构建ArrayList实例,以减少扩容次数,提高效率。

基本操作总结:

(1)增加元素的操作:

 增加单个元素的方法是add(E e)和add(int index, E e)。这两个方法都是向容器中添加新元素,这可能会导致capacity不足,因此在添加元素之前,都需要进行剩余空间检查,如果需要则自动扩容。

addAll()方法能够一次添加多个元素,根据位置不同也有两个版本,一个是在末尾添加的addAll(Collection<? extends E> c)方法,一个是从指定位置开始插入的addAll(int

index, Collection<? extends E> c)方法。跟add()方法类似,在插入之前也需要进行空间检查,如果需要则自动扩容;如果从指定位置插入,也会存在移动元素的情况。 addAll()的时间复杂度不仅跟插入元素的多少有关,也跟插入的位置相关。

添加元素时使用 ensureCapacityInternal() 方法来保证容量足够,如果不够时,需要使用 grow() 方法进行扩容,新容量的大小为 oldCapacity + (oldCapacity >> 1),也就是旧容量的 1.5 倍。

扩容操作需要调用 Arrays.copyOf() 把原数组整个复制到新数组中,这个操作代价很高,因此最好在创建 ArrayList 对象时就指定大概的容量大小,减少扩容操作的次数。

public booleanadd(Ee) {

    ensureCapacityInternal(size+ 1);  // Increments modCount!!

    elementData[size++] =e;

    return true;

}

private void ensureCapacityInternal(intminCapacity) {

    if (elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA) {

        minCapacity=Math.max(DEFAULT_CAPACITY, minCapacity);

    }

    ensureExplicitCapacity(minCapacity);

}

private void ensureExplicitCapacity(intminCapacity) {

    modCount++;

    // overflow-conscious code

    if (minCapacity - elementData.length > 0)

        grow(minCapacity);

}

private void grow(intminCapacity) {

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

}


小结:

先判断是否越界,是否需要扩容。

如果扩容, 就复制数组。

然后设置对应下标元素值。


值得注意的是:

1 如果需要扩容的话,默认扩容一半。如果扩容一半不够,就用目标的size作为扩容后的容量。

2 在扩容成功后,会修改modCount


(2)删除元素的操作:

    会修改modCount。remove()方法也有两个版本,一个是remove(int index)删除指定位置的元素,另一个是remove(Object o)删除第一个满足o.equals(elementData[index])的元素。删除操作是add()操作的逆过程,需要将删除点之后的元素向前移动一个位置。需要注意的是为了让GC起作用,必须显式的为最后一个位置赋null值。removeAll(Collection<?> c)方法批量删除一个列表的元素,用removeAll可以实现两个集合的差集。

需要调用 System.arraycopy() 将 index+1 后面的元素都复制到 index 位置上,该操作的时间复杂度为 O(N),可以看出 ArrayList 删除元素的代价是非常高的。

publicEremove(intindex) {

    rangeCheck(index);

    modCount++;

    EoldValue=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

    returnoldValue;

}



(3)修改元素的操作:

     不会修改modCount,相对增删是高效的操作。


public E set(int index, E element) {

    rangeCheck(index);//越界检查

    E oldValue =elementData(index); //取出元素

    elementData[index] =element;//覆盖元素

    return oldValue;//返回元素

}


(4)查找元素的操作:

     不会修改modCount,相对增删是高效的操作。


public E get(int index) {

    rangeCheck(index);//越界检查

    returnelementData(index); //下标取数据

}

E elementData(int index) {

    return (E)elementData[index];

}


(5)清空,clear

     会修改modCount。


public void clear() {

    modCount++;//修改modCount

    // clear to let GC do its work

    for (int i = 0; i

        elementData[i] =null;

    size = 0; //修改size

}


(6)包含contain

//普通的for循环寻找值,只不过会根据目标对象是否为null分别循环查找。

public boolean contains(Object o) {

    return indexOf(o) >=0;

}

//普通的for循环寻找值,只不过会根据目标对象是否为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;

}

增删改查中,增导致扩容,则会修改modCount,删一定会修改。改和查一定不会修改modCount。

扩容操作会导致数组复制,批量删除会导致 找出两个集合的交集,以及数组复制操作,因此,增、删都相对低效。而改、查都是很高效的操作。


Fail-fast:

fail-fast 机制,即快速失败机制,是java集合(Collection)中的一种错误检测机制。当在迭代集合的过程中该集合在结构上发生改变的时候,就有可能会发生fail-fast,即抛出ConcurrentModificationException异常。fail-fast机制并不保证在不同步的修改下一定会抛出异常,它只是尽最大努力去抛出,所以这种机制一般仅用于检测bug。

modCount 用来记录 ArrayList 结构发生变化的次数。结构发生变化是指添加或者删除至少一个元素的所有操作,或者是调整内部数组的大小,仅仅只是设置元素的值不算结构发生变化。

出现的场景:

在进行序列化或者迭代等操作时,需要比较操作前后 modCount 是否改变,如果改变了需要抛出 ConcurrentModificationException。


出现的原理:

     public static voidmain(String[] args) {

          List list = new ArrayList<>();

           for (int i = 0 ;i < 10 ; i++ ) {

                list.add(i +"");

           }

          Iterator iterator = list.iterator();

           int i = 0 ;

          while(iterator.hasNext()) {

                if (i == 3){

                    list.remove(3);

                }

               System.out.println(iterator.next());

                i ++;

           }

     }

看看最关心的next()方法,看看为什么在迭代过程中,如果有线程对集合结构做出改变,就会发生fail-fast:

      @SuppressWarnings("unchecked")

        public E next() {

            checkForComodification();

            int i = cursor;

            if (i >=size)

                throw newNoSuchElementException();

            Object[]elementData = ArrayList.this.elementData;

            if (i >=elementData.length)

                throw newConcurrentModificationException();

            cursor = i + 1;

            return (E)elementData[lastRet = i];

        }

从源码知道,每次调用next()方法,在实际访问元素前,都会调用checkForComodification方法,该方法源码如下:

        final voidcheckForComodification() {

            if (modCount !=expectedModCount)

                throw newConcurrentModificationException();

        }


可以看出,该方法才是判断是否抛出ConcurrentModificationException异常的关键。在该段代码中,当modCount != expectedModCount

时,就会抛出该异常。但是在一开始的时候,expectedModCount初始值默认等于modCount,为什么会出现modCount != expectedModCount,很明显expectedModCount在整个迭代过程除了一开始赋予初始值modCount外,并没有再发生改变,所以可能发生改变的就只有modCount,在前面关于ArrayList扩容机制的分析中,可以知道在ArrayList进行add,remove,clear等涉及到修改集合中的元素个数的操作时,modCount就会发生改变(modCount ++),所以当另一个线程(并发修改)或者同一个线程遍历过程中,调用相关方法使集合的个数发生改变,就会使modCount发生变化,这样在checkForComodification方法中就会抛出ConcurrentModificationException异常。


序列化时也可能发生fail-fast:

private voidwriteObject(java.io.ObjectOutputStreams)

    throwsjava.io.IOException{

    // Write out element count, and any hidden stuff

    int expectedModCount =modCount;

    s.defaultWriteObject();


    // Write out size as capacity for behavioural compatibility with clone()

    s.writeInt(size);


    // Write out all elements in the proper order.

    for (int i=0; i<size; i++) {

        s.writeObject(elementData[i]);

    }


    if (modCount !=expectedModCount) {

        throw newConcurrentModificationException();

    }

}


避免fail-fast:

方法1

在单线程的遍历过程中,如果要进行remove操作,可以调用迭代器的remove方法而不是集合类的remove方法。看看ArrayList中迭代器的remove方法的源码:

        public void remove(){

            if (lastRet <0)

                throw newIllegalStateException();

           checkForComodification();


            try {

               ArrayList.this.remove(lastRet);

                cursor =lastRet;

                lastRet =-1;

               expectedModCount = modCount;

            } catch(IndexOutOfBoundsException ex) {

                throw newConcurrentModificationException();

            }

        }

可以看到,该remove方法并不会修改modCount的值,并且不会对后面的遍历造成影响,因为该方法remove不能指定元素,只能remove当前遍历过的那个元素,所以调用该方法并不会发生fail-fast现象。该方法有局限性。


方法2

使用java并发包(java.util.concurrent)中的类来代替ArrayList 和hashMap。

比如使用 CopyOnWriterArrayList代替ArrayList,CopyOnWriterArrayList在是使用上跟ArrayList几乎一样,CopyOnWriter是写时复制的容器(COW),在读写时是线程安全的。该容器在对add和remove等操作时,并不是在原数组上进行修改,而是将原数组拷贝一份,在新数组上进行修改,待完成后,才将指向旧数组的引用指向新数组,所以对于CopyOnWriterArrayList在迭代过程并不会发生fail-fast现象。但 CopyOnWrite容器只能保证数据的最终一致性,不能保证数据的实时一致性。

对于HashMap,可以使用ConcurrentHashMap,ConcurrentHashMap采用了锁机制,是线程安全的。在迭代方面,ConcurrentHashMap使用了一种不同的迭代方式。在这种迭代方式中,当iterator被创建后集合再发生改变就不再是抛出ConcurrentModificationException,取而代之的是在改变时new新的数据从而不影响原有的数据 ,iterator完成后再将头指针替换为新的数据 ,这样iterator线程可以使用原来老的数据,而写线程也可以并发的完成改变。即迭代不会发生fail-fast,但不保证获取的是最新的数据。


替代方案:

可以使用 Collections.synchronizedList(); 得到一个线程安全的 ArrayList。

List list = new ArrayList<>();

List synList = Collections.synchronizedList(list);

也可以使用 concurrent 并发包下的 CopyOnWriteArrayList 类。

List list = new CopyOnWriteArrayList<>();


Vector

* Vector与其的区别在于Vector在API上都加了synchronized所以它是线程安全的,Vector 是同步的,因此开销就比 ArrayList 要大,访问速度更慢。最好使用 ArrayList 而不是 Vector,因为同步操作完全可以由程序员自己来控制;

* Vector扩容时,是翻倍size,而ArrayList是扩容到1.5倍。


2)LinkedList:

概括的说,LinkedList是线程不安全的,允许元素为null的双向链表。

其底层数据结构是双向链表,它实现List<E>, Deque<E>, Cloneable,

java.io.Serializable接口,它实现了Deque<E>,所以它也可以作为一个双端队列。和ArrayList比,没有实现RandomAccess所以其以下标,随机访问元素速度较慢。

因其底层数据结构是链表,所以可想而知,它的增删只需要移动指针即可,故时间效率较高。不需要批量扩容,也不需要预留空间,所以空间效率比ArrayList高。

缺点就是需要随机访问元素时,时间效率很低,虽然底层在根据下标查询Node的时候,会根据index判断目标Node在前半段还是后半段,然后决定是顺序还是逆序查询,以提升时间效率。不过随着n的增大,总体时间效率依然很低。


基本操作总结:

节点Node结构:

private static class Node {

    E item;//元素值

    Node next;//后置节点

    Node prev;//前置节点


    Node(Node prev,E element, Node next) {

        this.item = element;

        this.next = next;

        this.prev = prev;

    }

}

(1)增加元素的操作:

    增加单个元素的add()方法有两个版本,一个是add(E e),该方法在LinkedList的末尾插入元素,因为有last指向链表末尾,在末尾插入元素的花费是常数时间。只需要简单修改几个相关引用即可;另一个是add(int index, E element),该方法是在指定下表处插入元素,需要先通过线性查找找到具体位置,然后修改相关引用完成插入操作。

    批量增加元素的addAll()的方法:

    链表批量增加,是靠for循环遍历待插入数组,依次执行插入节点操作。对比Array

List是通过System.arraycopy完成批量增加的;

通过下标获取某个node的时候(增、查),会根据index处于前半段还是后半段进行一个折半,以提升查询效率。


(2)删除元素的操作:

remove()方法也有两个版本,一个是删除跟指定元素相等的第一个元素remove

(Object o),另一个是删除指定下标处的元素remove(int index)。

两个删除操作都要:1.先找到要删除元素的引用,2.修改相关引用,完成删除操作。

在寻找被删元素引用的时候remove(Object o)调用的是元素的equals方法,而remove(int index)使用的是下标计数,两种方式都是线性时间复杂度。在步骤2中,两个revome()方法都是通过unlink(Node<E> x)方法完成的。这里需要考虑删除元素是第一个或者最后一个时的边界情况。


总结:

链表批量增加,是靠for循环遍历待插入数组,依次执行插入节点操作。对比ArrayList是通过System.arraycopy完成批量增加的。增加一定会修改modCount;

通过下标获取某个node的时候(增、查),会根据index处于前半段还是后半段进行一个折半,以提升查询效率;

删也一定会修改modCount。按下标删,也是先根据index找到Node,然后去链表上unlink掉这个Node。按元素删,会先去遍历链表寻找是否有该Node,如果有,去链表上unlink掉这个Node;

改也是先根据index找到Node,然后替换值。改不修改modCount;

查本身就是根据index找到Node。


为什么Java里的Arrays.asList生成的List不能用addremove方法?

在平时的开发过程中,我们知道可以将一个Array的对象转化为List。这样的操作,我们只要采用Arrays.asList这个方法就行了。前段时间一直用这个方法,有一天,我发现通过Arrays.asList得到的List无法进行add和remove等操作。

下面是一段很简单的测试代码:

    public class MainFacade{

        public static voidmain(String[] args) {

           List list = Arrays.asList(1,2,3);

            list.add(5);

           System.out.print(list.toString());

        }

}

不过上面的代码会throw出一个UnsupportedOperationException这样的异常:

Exception in thread "main"java.lang.UnsupportedOperationException atjava.util.AbstractList.add(AbstractList.java:148) atjava.util.AbstractList.add(AbstractList.java:108) atorg.popkit.MainFacade.main(MainFacade.java:14) atsun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) atsun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)atsun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)at java.lang.reflect.Method.invoke(Method.java:606) atcom.intellij.rt.execution.application.AppMain.main(AppMain.java:134)

终其原因是Arrays.asList方法返回的ArrayList是继承自AbstractList同时实现了RandomAccess和Serializable接口,定义如下: 

    private static classArrayList extends AbstractList

        implementsRandomAccess, java.io.Serializable

我们再来看看AbstractList这个类的定义:

public abstract class AbstractList extendsAbstractCollection implements List

这时我们发现AbstractList这个类的set add remove方法定义如下:

    public void add(intindex, E element) {

        throw newUnsupportedOperationException();

    }


    public E set(int index,E element) {

        throw newUnsupportedOperationException();

    }


    public E remove(intindex) {

        throw newUnsupportedOperationException();

}

现在知道了它throw UnsupportedOperationException异常的原因了。 

通过上面的分析,我们知道,其实通过asList方法得到的List是只读的,那么平时我们怎样避免这样的错误发生?我们可以采用如下方法: 

List list = newArrayList<>(Arrays.asList(1,2,3));


为什么通过new生成的List可以正常执行add、remove等操作呢?

public class ArrayList extends AbstractList

        implementsList, RandomAccess, Cloneable, java.io.Serializable

因为在ArrayList中覆盖和重新实现了父类AbstractList中的add和remove等方法。

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

推荐阅读更多精彩内容