JavaSE学习——迭代器

java 复制代码
        List<String> list1 = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        for(String s : list1){
            System.out.println(s);
        }

迭代器相当于一种遍历的方式,专门用来遍历集合的工具。

三个核心方法:hasNext()判断是否存在下一个元素;next()返回下一个元素,remove删除刚遍历的那个元素。

比如下面的代码构建一个迭代器,用来遍历集合的元素:

java 复制代码
public class Main {
    public static void main(String[] args){
        List<String> list1 = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        Iterator<String> iterator = list1.iterator();
        while(iterator.hasNext()){
            String next = iterator.next();
            System.out.println(next);
        }
    }
}

迭代器只能用一次,比如:

java 复制代码
        List<String> list1 = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        Iterator<String> iterator = list1.iterator();
        while(iterator.hasNext()){
            String next = iterator.next();
            System.out.println(next);
        }
        while(iterator.hasNext()){
            String next = iterator.next();
            System.out.println(next);
        }

现在使用迭代器的remove方法,迭代之后进行删除:

java 复制代码
        List<String> list1 = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        Iterator<String> iterator = list1.iterator();
        while(iterator.hasNext()){
            String next = iterator.next();
            iterator.remove();
            System.out.println(next);
        }
        System.out.println(list1);

forEachRemaining()方法也可以进行遍历,传入的是一个consumer,有str就可以进行消费:(比如下面的例子就是进行一个打印输出)。

java 复制代码
        List<String> list1 = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        Iterator<String> iterator = list1.iterator();
        iterator.forEachRemaining(str->{
            System.out.println(str);
        });
        System.out.println(list1);

那回到我们最初的一个语法糖,它的底层就是迭代器:

for(String s : list1){

System.out.println(s);

}


比如现在还想打印一下下标:

java 复制代码
        List<String> list = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        int index = 0;
        for(String s: list){
            System.out.println(s+"位于"+index);
            index++;
        }

下面的代码也可以:(注意看index的定义位置)

java 复制代码
public class Main {
    static int index = 0;
    public static void main(String[] args){
        List<String> list = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));

        list.forEach(str->{
            System.out.println(str+"下标是:"+index);
            index++;
        });
    }
}

在lambda表达式里面不可以对局部变量进行一个改变,但是可以该表全局变量。下面的代码就是不可以的:

java 复制代码
public class Main {
    
    public static void main(String[] args){
        List<String> list = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        int index = 0;
        list.forEach(str->{
            System.out.println(str+"下标是:"+index);
            index++;
        });
    }
}

下面简单看一下顶层的接口父类 Iterable

创建了一个迭代器:

java 复制代码
Iterator<T> iterator();

默认实现了forEach()方法:

java 复制代码
    default void forEach(Consumer<? super T> action) {
        Objects.requireNonNull(action);
        for (T t : this) {
            action.accept(t);
        }
    }

此外,还有listIterator方法,是列表专用的迭代器方法,只要是列表都可以进行调用。相当于为列表专门强化了一遍迭代器。

java 复制代码
        List<String> list = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        ListIterator<String> stringListIterator = list.listIterator();

得到的对象可以调用更多的迭代器方法:

复制代码
比如previous()方法可以得到前一个元素,previousIndex()方法可以得到前一个元素的下标等等,还可以add()方法进行边遍历边添加。
java 复制代码
        List<String> list = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        ListIterator<String> stringListIterator = list.listIterator();
        while(stringListIterator.hasNext()){
            int i = stringListIterator.nextIndex();
            String next = stringListIterator.next();
            System.out.println("当前元素: "+next+", 位置: " + i);
        }

现在使用add()方法:

java 复制代码
        while(stringListIterator.hasNext()){
            int i = stringListIterator.nextIndex();
            String next = stringListIterator.next();
            stringListIterator.add("1314");
            System.out.println("当前元素: "+next+", 位置: " + i);
        }

现在我们知道,上面的迭代器代码运行完之后,迭代器的指针指向了最后一个元素,因为加强后的listIterator还可以验证前面的元素是否存在,所以可以向前遍历:

java 复制代码
        while(stringListIterator.hasNext()){
            int i = stringListIterator.nextIndex();
            String next = stringListIterator.next();
            stringListIterator.add("1314");
            System.out.println("当前元素: "+next+", 位置: " + i);
        }
        while (stringListIterator.hasPrevious()){
            int i = stringListIterator.previousIndex();
            String pre = stringListIterator.previous();
            System.out.println("当前元素:" + pre + "下标是:" + i);
        }

当然,因为这个迭代器是可以双向使用的,所以可以不止使用一次


有add()方法,也有remove()方法。注意,虽然可以一边遍历,一边添加/移除,但是迭代器依然遍历的指针指向原始的元素,不会把新添加的元素给遍历掉。

之所以要使用迭代器进行一边遍历一边添加/删除。但是在语法糖里面不行,会报错。下面的代码是错的:

java 复制代码
        List<String> list = new LinkedList<>(Arrays.asList("lhw", "hxl", "520"));
        for(String s : list){
            list.add("");
        }
相关推荐
Laurence2 小时前
C++ 引入第三方库(一):直接引入源文件
开发语言·c++·第三方库·添加·添加库·添加包·源文件
查古穆2 小时前
栈-有效的括号
java·数据结构·算法
kyriewen113 小时前
你点的“刷新”是假刷新?前端路由的瞒天过海术
开发语言·前端·javascript·ecmascript·html5
Java面试题总结3 小时前
Spring - Bean 生命周期
java·spring·rpc
硅基诗人3 小时前
每日一道面试题 10:synchronized 与 ReentrantLock 的核心区别及生产环境如何选型?
java
014-code3 小时前
String.intern() 到底干了什么
java·开发语言·面试
421!3 小时前
GPIO工作原理以及核心
开发语言·单片机·嵌入式硬件·学习
摇滚侠3 小时前
JAVA 项目教程《苍穹外卖-12》,微信小程序项目,前后端分离,从开发到部署
java·开发语言·vue.js·node.js
楚国的小隐士4 小时前
为什么说Rust是对自闭症谱系人士友好的编程语言?
java·rust·编程·对比·自闭症·自闭症谱系障碍·神经多样性