Stream.forEach和Collection.forEach

本文详细介绍了Java中Collection与Stream的forEach方法的区别,特别是执行顺序和并行流的影响。在集合上直接调用forEach会按照插入顺序处理元素,而Stream.forEach对于顺序流不保证顺序,对于并行流则可能出现无序。同时,通过自定义迭代器可以改变元素遍历的顺序,但stream().forEach将忽略自定义迭代器。示例代码展示了不同forEach方法的行为差异。
摘要由CSDN通过智能技术生成


一、定义

forEach由 Iterable.forEach 定义,而stream.forEach 由 Stream.forEach 定义。 Stream.forEach 的定义允许以任何顺序处理元素 - 即使对于顺序流也是如此。 但对于并行流, Stream.forEach 很可能会无序处理元素。

Iterable.forEach 从源获取一个Iterator并在其上调用 forEachRemaining()。
Stream.forEach 的实现都将创建一个源自其中一个迭代器的Spliterator,然后在Iterator上调用 forEachRemaining

二、执行顺序

Collection.forEach()使用集合的迭代器(如果指定了一个),集合里元素的处理顺序是明确的。相反,Collection.stream().forEach()的处理顺序是不明确的。

1、Parallel Stream

    public void test29() {
        List<String> list = Arrays.asList("A","B","C","D");
        list.forEach(System.out::print);
        System.out.println("=================================================");
        list.stream().forEach(System.out::print);
        System.out.println("=================================================");
        list.parallelStream().forEach(System.out::print);
        System.out.println("=================================================");
    }

我们会看到 list.forEach()以插入顺序处理元素,而 list.parallelStream().forEach()在每次运行会产生不同的结果。一个可能的输出是:
在这里插入图片描述

三、自定义迭代器

/**
 * @author lichangyuan
 * @create 2022-03-03 16:05
 */
public class ReverseList extends ArrayList<String> {

    @Override
    public Iterator<String> iterator() {

        int startlndex = this.size() - 1;
        List<String> list = this;

        Iterator<String> it = new Iterator<String>() {

            private int currentlndex = startlndex;

            @Override
            public boolean hasNext() {
                return currentlndex >= 0;
            }

            @Override
            public String next() {
                String next = list.get(currentlndex);
                currentlndex--;
                return next;
            }

            @Override
            public void remove() {
                throw new UnsupportedOperationException();
            }
        };
        return it;
    }
}
    @Test
    public void test30() {
        List<String> list = Arrays.asList("A","B","C","D");
        List<String> myList = new ReverseList();
        myList.addAll(list);

        myList.forEach(System.out::print);
        System.out.println("=================================================");
        myList.stream().forEach(System.out::print);
        System.out.println("=================================================");
        myList.parallelStream().forEach(System.out::print);
        System.out.println("=================================================");
    }

结果不同的原因是在列表中使用的 forEach()会使用自定义迭代器,而 stream().forEach()只是从列表中逐个获取元素,会忽略迭代器。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

和烨

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值