Java Stream过滤实现原理详解

一、Stream简介

Stream是Java 8中引入的一种全新的函数式编程方式,可以看做对集合和数组操作的函数化封装。

Stream可以进行过滤、映射、统计、归约等多种操作,可以大大简化代码复杂度,提高编程效率。

 // Stream是java.util.stream包下的类,需要进行导包  
import java.util.stream.Stream;

二、Stream过滤方法filter()

Stream中的filter方法用于过滤数据,筛选出符合条件的元素。

filter方法接受一个Predicate接口类型的lambda表达式,该表达式用于确定元素是否符合过滤条件。

 // 过滤出大于等于10的数字  
Stream<Integer> stream1 = Stream.of(5, 10, 15, 20);  
Stream<Integer> stream2 = stream1.filter(i -> i >= 10);  
stream2.forEach(System.out::println);  
// 输出:  
// 10  
// 15  
// 20  

三、Stream过滤实现原理

filter方法实现过滤的关键在于,判断每个元素是否符合条件,只输出符合条件的元素。

实现这个过程需要满足两个条件:

  • 每个元素都要被判断,即需要用到遍历方法。
  • 满足筛选条件的元素需要输出,即需要用到输出方法。

Stream本质上就是对这两个条件进行封装,核心就是使用map、reduce等方法实现中间操作,最终使用forEach等方法实现终端操作。

具体流程如下:

  1. 获取数据源——Stream.of或集合的stream方法等。
  2. 进行中间操作——如filter方法,进行过滤。
  3. 进行终端操作——如forEach方法,输出符合条件的元素。

四、Stream过滤实现原理示例

通过对Stream源码的分析,可以看出Stream的具体实现原理。

1. Stream的准备

Stream的创建方式有多种,示例中使用of方法创建一个int类型的Stream:

Stream<Integer> stream = Stream.of(2, 3, 4, 5, 6);

2. filter方法实现

filter方法需要传入一个Predicate类型的lambda表达式,用于判断元素是否符合过滤条件。

具体实现过程如下:

public interface Predicate<T> {  
    //进行元素判断,判断是否符合过滤条件  
    boolean test(T t);  
}  

public interface Stream<T> extends BaseStream<T, Stream<T>> {  
    // 进行中间操作,如filter方法 
    Stream<T> filter(Predicate<? super T> predicate);  
}  

// 中间操作的具体实现    
@Override  
public final Stream<P_OUT> filter(Predicate<? super P_OUT> predicate) {  
    Objects.requireNonNull(predicate);  
    return new StatelessOp<>(this, StreamShape.REFERENCE, StreamOpFlag.NOT_SIZED) {  
        @Override  
        public Sink<P_OUT> opWrapSink(int flags, Sink<T> sink) {  
            return new Sink.ChainedReference<>(sink) {  
                @Override  
                public void begin(long size) {  
                    downstream.begin(-1);  
                }  
                @Override  
                public void accept(T t) {  
                    if (predicate.test(t)) {  
                        downstream.accept(t);  
                    }  
                }  
            };  
        }  
    };  
}

可以看出filter方法返回了一个新的Stream对象,其中StatelessOp实现了对原Stream的封装(对外提供的接口还是Stream)。

具体实现了两个方法:

  • opIsStateful:用于判断Stream是否能重复使用,filter方法中实现为return false。
  • opWrapSink:用于返回一个Sink对象,Sink是对元素遍历输出的封装。

opWrapSink返回的是一个匿名类,用于判断元素是否符合过滤条件:

if (predicate.test(t)) {  
    downstream.accept(t);  
}

3. forEach方法实现

forEach用于对Stream的符合条件的元素进行输出,具体实现过程如下:

public interface Consumer<T> {  
    void accept(T t);  
}  

public interface Stream<T> extends BaseStream<T, Stream<T>> {  
    // 对符合条件的元素进行逐个操作  
    void forEach(Consumer<? super T> action);  
}  

// 终端操作的具体实现  
@Override  
public final void forEach(Consumer<? super P_OUT> action) {  
    evaluate(ForEachOps.makeRef(action, !isParallel()));  
}

可以看出,forEach方法实际调用了evaluate方法。

evaluate方法的具体实现在PipelineHelper类中,对StatelessOp对象逐个进行输出操作。

下面是PipelineHelper的部分代码实现:

 @Override  
        public <E_OUT> TerminalOp<P_OUT, E_OUT> evaluate(TerminalOp<E_OUT, R> terminalOp) {  
            Objects.requireNonNull(terminalOp);  
            if (linkedOrConsumed) {  
                throw new IllegalStateException(MSG_STREAM_LINKED);  
            }  
            linkedOrConsumed = true;  
            return new ChainedTerminal<>(terminalOp).evaluateSequential(this);  
        }

static final class ChainedTerminal<E_IN, E_OUT>  
            implements TerminalOp<E_IN, E_OUT> {  
            private final TerminalOp<?, E_OUT> terminalOp;  

            ChainedTerminal(TerminalOp<?, E_OUT> terminalOp) {  
                this.terminalOp = terminalOp;  
            }  

            @Override  
            public StreamShape inputShape() {  
                return terminalOp.inputShape();  
            }  

            @Override  
            public <S> E_OUT evaluateSequential(PipelineHelper<S> helper, Spliterator<S> spliterator) {  
                return terminalOp.evaluateSequential(helper, ops.wrapSink(terminalOp, helper.wrapSink(terminalOp, terminalSink)  
                ));  
            }  
    }

@Override  
    // Input(depth==0), Output(Vector,X)  
    public Sink<T> wrapSink(PipelineHelper<T> ph, Sink<X> sink) {  
        return opWrapSink(opFlags, sink);  
    }  

五、总结

Java Stream过滤实现原理是在中间操作filter和终端操作forEach的配合下完成的。

Stream的本质是对集合和数组操作的封装,并不会增加实际的存储压力。

Stream的使用可以大大简化代码复杂度,提高编程效率。

下面是完整的示例代码:

import java.util.Objects;  
import java.util.function.Consumer;  
import java.util.function.Predicate;  
import java.util.stream.Stream;  

public class StreamFilterDemo {  
    public static void main(String[] args) {  
        // Stream的创建  
        Stream<Integer> stream = Stream.of(2, 3, 4, 5, 6);  
        // 过滤大于等于4的元素  
        Stream<Integer> stream2 = stream.filter(i -> i >= 4);  
        // 输出符合条件的元素  
        stream2.forEach(System.out::println);  
    }  
}  

原创文章,作者:小蓝,如若转载,请注明出处:https://www.506064.com/n/245531.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
小蓝的头像小蓝
上一篇 2024-12-12 13:09
下一篇 2024-12-12 13:09

相关推荐

  • Java JsonPath 效率优化指南

    本篇文章将深入探讨Java JsonPath的效率问题,并提供一些优化方案。 一、JsonPath 简介 JsonPath是一个可用于从JSON数据中获取信息的库。它提供了一种DS…

    编程 2025-04-29
  • java client.getacsresponse 编译报错解决方法

    java client.getacsresponse 编译报错是Java编程过程中常见的错误,常见的原因是代码的语法错误、类库依赖问题和编译环境的配置问题。下面将从多个方面进行分析…

    编程 2025-04-29
  • Java Bean加载过程

    Java Bean加载过程涉及到类加载器、反射机制和Java虚拟机的执行过程。在本文中,将从这三个方面详细阐述Java Bean加载的过程。 一、类加载器 类加载器是Java虚拟机…

    编程 2025-04-29
  • Java腾讯云音视频对接

    本文旨在从多个方面详细阐述Java腾讯云音视频对接,提供完整的代码示例。 一、腾讯云音视频介绍 腾讯云音视频服务(Cloud Tencent Real-Time Communica…

    编程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介绍

    本文将详细介绍Java Milvus SearchParam withoutFields的相关知识和用法。 一、什么是Java Milvus SearchParam without…

    编程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java语言中的一个版本,于2014年3月18日发布。本文将从多个方面对Java 8中某一周的周一进行详细的阐述。 一、数组处理 Java 8新特性之一是Stream…

    编程 2025-04-29
  • Java判断字符串是否存在多个

    本文将从以下几个方面详细阐述如何使用Java判断一个字符串中是否存在多个指定字符: 一、字符串遍历 字符串是Java编程中非常重要的一种数据类型。要判断字符串中是否存在多个指定字符…

    编程 2025-04-29
  • VSCode为什么无法运行Java

    解答:VSCode无法运行Java是因为默认情况下,VSCode并没有集成Java运行环境,需要手动添加Java运行环境或安装相关插件才能实现Java代码的编写、调试和运行。 一、…

    编程 2025-04-29
  • Java任务下发回滚系统的设计与实现

    本文将介绍一个Java任务下发回滚系统的设计与实现。该系统可以用于执行复杂的任务,包括可回滚的任务,及时恢复任务失败前的状态。系统使用Java语言进行开发,可以支持多种类型的任务。…

    编程 2025-04-29
  • Harris角点检测算法原理与实现

    本文将从多个方面对Harris角点检测算法进行详细的阐述,包括算法原理、实现步骤、代码实现等。 一、Harris角点检测算法原理 Harris角点检测算法是一种经典的计算机视觉算法…

    编程 2025-04-29

发表回复

登录后才能评论