java基础(5)-Iterator

1
2
3
    for (StringBuilder s : list) {
        // do something
    }

这样类似的代码,应该大家都写过无数次了。这是foreach的语法, 可以看这篇文章Java语法糖1:可变长度参数以及foreach循环原理。该文中讲到,集合使用foreach会去调用它的迭代器(iterator),数组调用时就是调用它的自身的循环

要获取iterator,必须要实现Iterable接口

Implementing this interface allows an object to be the target of the “for-each loop” statement. – jdk

该接口挺简单的就不在这里多讲。

当阅读集合源码的时候,会时不时看见ListIterator这个迭代器

ListIterator(列表迭代器)

ListIterator允许使用者在任意方向(正、反向)进行遍历列表,在迭代过程中修改列表,并获取迭代器在列表中的当前位置

但是在迭代过程中修改列表,可能会报 UnsupportedOperationException,当然要看具体的列表类,是否实现了修改操作。

如果需要使用到,反向遍历或在迭代期间做修改操作的话,可以考虑使用列表迭代器

Spliterator(分离器)

An object for traversing and partitioning elements of a source –jdk

java8引入的,用与遍历和划分源元素的对象。Spliterator,象Iterator,用来遍历源中的元素。

Spliterator被设计来最大的特点就是,支持有效的并行遍历(通过分解以及单元素迭代),除了顺序遍历之外。

测试并行遍历代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
        AtomicInteger index = new AtomicInteger();
        List<StringBuilder> list = Stream.generate(() -> new StringBuilder("hello world!" + index.getAndIncrement()))
                .limit(1000)
                .collect(Collectors.toList());

        Spliterator<StringBuilder> s1 = list.spliterator();
        //分成几份,互相不影响 trySplit()
        Spliterator<StringBuilder> s2 = s1.trySplit();
        // 返回s1、s2的估计值 
        System.out.println(s2.estimateSize() + " " + s1.estimateSize());

        ExecutorService es = Executors.newFixedThreadPool(10);
        es.execute(() -> {
          // tryAdvance(Consumer<? super T> action) 元素存在就执行,相应的操作
            while(s1.tryAdvance(System.out::println))
            {
                System.out.println(Thread.currentThread().getName());
                //                    Thread.sleep(50);
            };
        });

        es.execute(() -> {
            while (s2.tryAdvance(System.out::println)) {
                System.out.println(Thread.currentThread().getName());
                //                    Thread.sleep(50);
            }
        });
        es.shutdown();

// output:
// 500 500
// hello world!0
// pool-1-thread-2
// hello world!1
// pool-1-thread-2
// hello world!2
// ....
// hello world!504
// hello world!11
// pool-1-thread-2
// pool-1-thread-1
// hello world!505
// ...

参考: 1. Spliterator-oracle


对自己的现状不满意只有付出更多的努力去改变它

如果有不对的地方或建议,请指出,谢谢啦