JAVA语言之全面解析ListIterator 与 Iterator 异同
从安 2019-06-05 来源 : 阅读 892 评论 0

摘要:本篇文章主要讲述JAVA语言之全面解析ListIterator 与 Iterator 异同,希望阅读本篇文章以后大家有所收获,帮助大家对相关内容的理解更加深入。

本篇文章主要讲述JAVA语言之全面解析ListIterator 与 Iterator 异同,希望阅读本篇文章以后大家有所收获,帮助大家对相关内容的理解更加深入。

JAVA语言之全面解析ListIterator 与 Iterator 异同

一、概述

基于fail-fast机制,我们知道对于ArrayList等集合在迭代过程中是不可进行结构修改操作的,唯一能使用的结构修改操作只有Iterator接口中的remove()方法。

而java.util.ListIterator接口继承自Iterator接口,是专用于列表结构集合的迭代器,在 Iterator 的基础上,额外提供了 previous、nextIndex、add、set 等方法。

next、previous语义上的小问题

迭代器中的next()方法不应该理解为返回下一个元素,而是返回当前元素,并将指针移向下一个。

cursor永远指向下一个待返回的元素下标。

previous() 则是将指针向前移动一个位置后返回元素。

参考如下情况:

    // 对于 List{3,4,5}
    ListIterator listIterator = list.listIterator();
    boolean isDo = false; //isDo保证只向前移动一次,否则会死循环
    while(listIterator.hasNext()){
        Integer i = listIterator.next();
        if(i.equals(4)&&!isDo){
            i = listIterator.previous();
            isDo = true;
        }
        System.out.println(i);
    }

最后输出:

    3
    4 //因为equal(4)时指针已经指向5了, previous会导致指针重回4
       //(而不是字面上想的当==4时,返回上一个3)
    4
    5

修改对迭代过程是否可见?

是否是可见具体看类(暂未找到对比)。如 对于ArrayList来说,迭代器并未使用副本数组,因此修改是可见的(但需要一定操作,因为add方法添加元素后会把指针再往后移一位【即,若一直next的话,等于忽略了迭代过程中添加的元素】)

二、一些类对LisIterator实现详解

2.1 ArrayList 对 ListIterator 的实现

对于ArrayList对ListIterator的实现来说,并不是取消了fail-fast机制,而是调用迭代器来修改的话,每次修改后都令expectedModCount = modCount,因此不会报ConcurrentModificationException异常。

源码如下:ListItr是ArrayList的内部类

    private class ListItr extends Itr implements ListIterator {
        ListItr(int index) {
            super();
            cursor = index;  //构建是可指定迭代开始下标,默认是0
        }
        public boolean hasPrevious() { return cursor != 0; }
        /** 因为cursor始终指向下个元素下标 */
        public int nextIndex() { return cursor; }
        public int previousIndex() { return cursor - 1; }
        /** 返回cursor-1的元素 */
        @SuppressWarnings("unchecked")
        public E previous() {
            checkForComodification();
            int i = cursor - 1;
            if (i < 0)
                throw new NoSuchElementException();
            Object[] elementData = ArrayList.this.elementData;
            if (i >= elementData.length)
                throw new ConcurrentModificationException();
            cursor = i;
            return (E) elementData[lastRet = i];   //lastRet始终保存上次获取元素的下标
        }
        /** 省略set源码,set方法用以设置lastRet元素 */
        /** 添加方法*/
        public void add(E e) {
            checkForComodification();
            try {
                int i = cursor;
                ArrayList.this.add(i, e); //将元素添加在当前元素后面,也就是cursor的位置
                cursor = i + 1; //再将cursor+1,等于忽略刚添加的元素
                lastRet = -1;
                expectedModCount = modCount;
            } catch (IndexOutOfBoundsException ex) {
                throw new ConcurrentModificationException();
            }
        }
    }

2.2 CopyOnWriteArrayList 对ListIterator的实现

对于CopyOnWriteArrayList,虽然有lisIterator方法,但实则不支持列表迭代器的结构修改方法,如add、remove等(直接抛UnsupportedOperationException),因为它 fail-safe机制本身就支持迭代过程中去修改集合。但修改是不可见的,具体参考CopyOnWriteArrayList对fail-safe实现


本文由职坐标整理发布,学习更多的相关知识,请关注职坐标IT知识库!


本文由 @从安 发布于职坐标。未经许可,禁止转载。
喜欢 | 0 不喜欢 | 0
看完这篇文章有何感觉?已经有0人表态,0%的人喜欢 快给朋友分享吧~
评论(0)
后参与评论

您输入的评论内容中包含违禁敏感词

我知道了

助您圆梦职场 匹配合适岗位
验证码手机号,获得海同独家IT培训资料
选择就业方向:
人工智能物联网
大数据开发/分析
人工智能Python
Java全栈开发
WEB前端+H5

请输入正确的手机号码

请输入正确的验证码

获取验证码

您今天的短信下发次数太多了,明天再试试吧!

提交

我们会在第一时间安排职业规划师联系您!

您也可以联系我们的职业规划师咨询:

小职老师的微信号:z_zhizuobiao
小职老师的微信号:z_zhizuobiao

版权所有 职坐标-一站式IT培训就业服务领导者 沪ICP备13042190号-4
上海海同信息科技有限公司 Copyright ©2015 www.zhizuobiao.com,All Rights Reserved.
 沪公网安备 31011502005948号    

©2015 www.zhizuobiao.com All Rights Reserved

208小时内训课程