> Java > java지도 시간 > Java Stream API에서 터미널 동작 예시 분석

Java Stream API에서 터미널 동작 예시 분석

王林
풀어 주다: 2023-05-08 17:34:17
앞으로
940명이 탐색했습니다.

    1. Java Stream 파이프라인 데이터 처리 작업

    이번 호 이전에 쓴 글에서 Java Stream 파이프라인 스트림은 컬렉션 클래스 요소의 처리를 단순화하는 데 사용되는 Java API라고 소개한 적이 있습니다. 사용과정은 3단계로 나누어집니다. 이 기사를 시작하기 전에 그림에 표시된 대로 이 세 단계를 새로운 친구들에게 소개해야 한다고 생각합니다.

    Java Stream API에서 터미널 동작 예시 분석

    • 첫 번째 단계(그림의 파란색): 컬렉션, 배열 또는 라인 변환 텍스트 파일 Java Stream 파이프라인 흐름

    • 의 두 번째 단계(그림에서 점선 부분): 파이프라인 스트리밍 데이터 처리 작업, 파이프라인의 각 요소를 처리합니다. 이전 파이프의 출력 요소는 다음 파이프의 입력 요소로 사용됩니다.

    • 세 번째 단계(그림에서 녹색): 이 글의 핵심 내용인 파이프라인 흐름 결과 처리 작업입니다.

    학습을 시작하기 전에 이전에 말씀드린 예제를 검토해야 합니다.

    List<String> nameStrs = Arrays.asList("Monkey", "Lion", "Giraffe","Lemur");
    List<String> list = nameStrs.stream()
            .filter(s -> s.startsWith("L"))
            .map(String::toUpperCase)
            .sorted()
            .collect(toList());
    System.out.println(list);
    로그인 후 복사

    먼저 stream() 메서드를 사용하여 문자열 List를 파이프라인 스트림 Stream으로 변환합니다

    그런 다음 파이프라인 데이터 처리 작업을 수행합니다. , 먼저 필터 함수를 사용하여 대문자 L로 시작하는 모든 문자열을 필터링한 다음 파이프라인의 문자열을 대문자 toUpperCase로 변환한 다음 sorted 메서드를 호출하여 정렬합니다. 이러한 API의 사용법은 이 문서의 이전 문서에서 소개되었습니다. 람다 표현식과 함수 참조도 사용됩니다.

    마지막으로 결과 처리를 위해 수집 기능을 사용하여 Java Stream 파이프라인 스트림을 목록으로 변환합니다. 목록의 최종 출력은 다음과 같습니다. [LEMUR, LION][LEMUR, LION]

    如果你不使用java Stream管道流的话,想一想你需要多少行代码完成上面的功能呢?回到正题,这篇文章就是要给大家介绍第三阶段:对管道流处理结果都可以做哪些操作呢?下面开始吧!

    二、ForEach和ForEachOrdered

    如果我们只是希望将Stream管道流的处理结果打印出来,而不是进行类型转换,我们就可以使用forEach()方法或forEachOrdered()方法。

    Stream.of("Monkey", "Lion", "Giraffe", "Lemur", "Lion")
            .parallel()
            .forEach(System.out::println);
    Stream.of("Monkey", "Lion", "Giraffe", "Lemur", "Lion")
            .parallel()
            .forEachOrdered(System.out::println);
    로그인 후 복사

    parallel()函数表示对管道中的元素进行并行处理,而不是串行处理,这样处理速度更快。但是这样就有可能导致管道流中后面的元素先处理,前面的元素后处理,也就是元素的顺序无法保证

    forEachOrdered从名字上看就可以理解,虽然在数据处理顺序上可能无法保障,但是forEachOrdered方法可以在元素输出的顺序上保证与元素进入管道流的顺序一致。也就是下面的样子(forEach方法则无法保证这个顺序):

    Monkey
    Lion
    Giraffe
    Lemur
    Lion

    三、元素的收集collect

    java Stream 最常见的用法就是:一将集合类转换成管道流,二对管道流数据处理,三将管道流处理结果在转换成集合类。那么collect()方法就为我们提供了这样的功能:将管道流处理结果在转换成集合类。

    3.1.收集为Set

    通过Collectors.toSet()方法收集Stream的处理结果,将所有元素收集到Set集合中。

    Set<String> collectToSet = Stream.of(
       "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    ) 
    .collect(Collectors.toSet());
    //最终collectToSet 中的元素是:[Monkey, Lion, Giraffe, Lemur],注意Set会去重。
    로그인 후 복사

    3.2.收集到List

    同样,可以将元素收集到List使用toList()收集器中。

    List<String> collectToList = Stream.of(
       "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    ).collect(Collectors.toList());
    
    // 最终collectToList中的元素是: [Monkey, Lion, Giraffe, Lemur, Lion]
    로그인 후 복사

    3.3.通用的收集方式

    上面为大家介绍的元素收集方式,都是专用的。比如使用Collectors.toSet()收集为Set类型集合;使用Collectors.toList()收集为List类型集合。那么,有没有一种比较通用的数据元素收集方式,将数据收集为任意的Collection接口子类型。 所以,这里就像大家介绍一种通用的元素收集方式,你可以将数据元素收集到任意的Collection类型:即向所需Collection类型提供构造函数的方式。

    LinkedList<String> collectToCollection = Stream.of(
       "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    ).collect(Collectors.toCollection(LinkedList::new));
    //最终collectToCollection中的元素是: [Monkey, Lion, Giraffe, Lemur, Lion]
    로그인 후 복사

    注意:代码中使用了LinkedList::new,实际是调用LinkedList的构造函数,将元素收集到Linked List。当然你还可以使用诸如LinkedHashSet::newPriorityQueue::new将数据元素收集为其他的集合类型,这样就比较通用了。

    3.4.收集到Array

    通过toArray(String[]::new)方法收集Stream的处理结果,将所有元素收集到字符串数组中。

    String[] toArray = Stream.of(
       "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    ) .toArray(String[]::new);
    //最终toArray字符串数组中的元素是: [Monkey, Lion, Giraffe, Lemur, Lion]
    로그인 후 복사

    3.5.收集到Map

    使用Collectors.toMap()方法将数据元素收集到Map里面,但是出现一个问题:那就是管道中的元素是作为key,还是作为value。我们用到了一个Function.identity()方法,该方法很简单就是返回一个“ t -> t ”(输入就是输出的lambda表达式)。另外使用管道流处理函数distinct()

    Java Stream 파이프라인 흐름을 사용하지 않는 경우 위 기능을 완료하는 데 몇 줄의 코드가 필요한지 생각해 보세요. 주제로 돌아가서 이 문서에서는 세 번째 단계인 파이프라인 스트림 처리 결과에 대해 어떤 작업을 수행할 수 있는지 소개합니다. 시작하자!

    2. ForEach 및 ForEachOrdered

    유형 변환을 수행하는 대신 Stream 파이프라인의 처리 결과만 인쇄하려면 forEach() 메서드 또는 forEachOrdered() 메서드를 사용할 수 있습니다. 🎜
    Map<String, Integer> toMap = Stream.of(
        "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    )
    .distinct()
    .collect(Collectors.toMap(
           Function.identity(),   //元素输入就是输出,作为key
           s -> (int) s.chars().distinct().count()// 输入元素的不同的字母个数,作为value
    ));
    // 最终toMap的结果是: {Monkey=6, Lion=4, Lemur=5, Giraffe=6}
    로그인 후 복사
    🎜parallel() 함수는 파이프라인의 요소가 직렬이 아닌 병렬로 처리되어 처리 속도가 더 빠르다는 것을 나타냅니다. 그러나 이로 인해 파이프라인 흐름의 나중 요소가 먼저 처리되고 이전 요소가 나중에 처리될 수 있습니다. 즉, 요소의 순서를 보장할 수 없습니다.🎜🎜forEachOrdered는 이름에서 이해할 수 있지만 데이터는 처리 순서는 보장되지 않을 수 있지만 forEachOrdered 메서드는 요소가 출력되는 순서가 요소가 파이프라인 스트림에 들어가는 순서와 일치하도록 보장할 수 있습니다. 이것이 아래와 같습니다(forEach 메소드는 이 순서를 보장할 수 없습니다): 🎜
    🎜Monkey
    Lion
    Giraffe
    Lemur
    Lion🎜
    🎜Three elementscollect🎜🎜java Stream의 가장 일반적인 사용법은 다음과 같습니다. 첫째, 컬렉션 클래스를 파이프라인 스트림으로 변환하고, 둘째, 파이프라인 스트림 데이터를 처리하고, 셋째, 파이프라인 스트림 처리 결과를 컬렉션 클래스로 변환합니다. 그런 다음 Collect() 메서드는 파이프라인 스트림 처리 결과를 컬렉션 클래스로 변환하는 기능을 제공합니다. 🎜🎜3.1. Collect as Set🎜🎜 Collectors.toSet() 메서드를 통해 Stream의 처리 결과를 수집하고 모든 요소를 ​​Set 컬렉션으로 수집합니다. 🎜
    Map<Character, List<String>> groupingByList =  Stream.of(
        "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    )
    .collect(Collectors.groupingBy(
           s -> s.charAt(0) ,  //根据元素首字母分组,相同的在一组
           // counting()        // 加上这一行代码可以实现分组统计
    ));
    // 最终groupingByList内的元素: {G=[Giraffe], L=[Lion, Lemur, Lion], M=[Monkey]}
    //如果加上counting() ,结果是:  {G=1, L=3, M=1}
    로그인 후 복사
    로그인 후 복사
    🎜3.2. List로 수집🎜🎜마찬가지로 toList() 수집기를 사용하여 요소를 List로 수집할 수 있습니다. 🎜
    boolean containsTwo = IntStream.of(1, 2, 3).anyMatch(i -> i == 2);
    // 判断管道中是否包含2,结果是: true
    long nrOfAnimals = Stream.of(
        "Monkey", "Lion", "Giraffe", "Lemur"
    ).count();
    // 管道中元素数据总计结果nrOfAnimals: 4
    int sum = IntStream.of(1, 2, 3).sum();
    // 管道中元素数据累加结果sum: 6
    OptionalDouble average = IntStream.of(1, 2, 3).average();
    //管道中元素数据平均值average: OptionalDouble[2.0]
    int max = IntStream.of(1, 2, 3).max().orElse(0);
    //管道中元素数据最大值max: 3
    IntSummaryStatistics statistics = IntStream.of(1, 2, 3).summaryStatistics();
    // 全面的统计结果statistics: IntSummaryStatistics{count=3, sum=6, min=1, average=2.000000, max=3}
    로그인 후 복사
    로그인 후 복사
    🎜3.3.범용 수집 방법🎜🎜위에 소개된 요소 수집 방법은 모두 전용입니다. 예를 들어, Set 유형 컬렉션을 수집하려면 Collectors.toSet()를 사용하고, List 유형 컬렉션을 수집하려면 Collectors.toList()를 사용하세요. 따라서 데이터 요소를 수집하여 Collection 인터페이스의 하위 유형으로 데이터를 수집하는 보다 일반적인 방법이 있습니까? 따라서 요소를 수집하는 일반적인 방법은 다음과 같습니다. 즉, 필수 컬렉션 유형에 생성자를 제공하여 데이터 요소를 컬렉션 유형으로 수집할 수 있습니다. 🎜rrreee🎜참고: LinkedList::new는 코드에서 실제로 LinkedList의 생성자를 호출하여 요소를 Linked List로 수집하는 데 사용됩니다. 물론 LinkedHashSet::newPriorityQueue::new와 같은 메서드를 사용하여 데이터 요소를 다른 컬렉션 유형으로 수집할 수도 있으며 이는 보다 다양한 용도로 사용할 수 있습니다. 🎜🎜3.4. Array로 수집🎜🎜 toArray(String[]::new) 메소드를 통해 Stream의 처리 결과를 수집하고 모든 요소를 ​​문자열 배열로 수집합니다. 🎜rrreee🎜3.5. Map으로 수집🎜🎜 Collectors.toMap() 메서드를 사용하여 데이터 요소를 Map으로 수집하지만 문제가 있습니다. 파이프라인의 요소가 키로 사용되는지 아니면 값으로 사용되는지입니다. 우리는 단순히 "t -> t"를 반환하는 Function.identity() 메서드를 사용했습니다(입력은 출력의 람다 표현식임). 또한 파이프라인 스트림 처리 함수 distinct()를 사용하여 맵 키 값의 고유성을 보장합니다. 🎜rrreee🎜3.6. GroupingBy🎜🎜Collectors.groupingBy는 요소의 그룹화 컬렉션을 구현하는 데 사용됩니다. 다음 코드는 첫 글자를 기반으로 다양한 데이터 요소를 다양한 목록으로 수집하고 이를 지도로 캡슐화하는 방법을 보여줍니다. 🎜
    Map<Character, List<String>> groupingByList =  Stream.of(
        "Monkey", "Lion", "Giraffe", "Lemur", "Lion"
    )
    .collect(Collectors.groupingBy(
           s -> s.charAt(0) ,  //根据元素首字母分组,相同的在一组
           // counting()        // 加上这一行代码可以实现分组统计
    ));
    // 最终groupingByList内的元素: {G=[Giraffe], L=[Lion, Lemur, Lion], M=[Monkey]}
    //如果加上counting() ,结果是:  {G=1, L=3, M=1}
    로그인 후 복사
    로그인 후 복사

    这是该过程的说明:groupingBy第一个参数作为分组条件,第二个参数是子收集器。

    四、其他常用方法

    boolean containsTwo = IntStream.of(1, 2, 3).anyMatch(i -> i == 2);
    // 判断管道中是否包含2,结果是: true
    long nrOfAnimals = Stream.of(
        "Monkey", "Lion", "Giraffe", "Lemur"
    ).count();
    // 管道中元素数据总计结果nrOfAnimals: 4
    int sum = IntStream.of(1, 2, 3).sum();
    // 管道中元素数据累加结果sum: 6
    OptionalDouble average = IntStream.of(1, 2, 3).average();
    //管道中元素数据平均值average: OptionalDouble[2.0]
    int max = IntStream.of(1, 2, 3).max().orElse(0);
    //管道中元素数据最大值max: 3
    IntSummaryStatistics statistics = IntStream.of(1, 2, 3).summaryStatistics();
    // 全面的统计结果statistics: IntSummaryStatistics{count=3, sum=6, min=1, average=2.000000, max=3}
    로그인 후 복사
    로그인 후 복사

    위 내용은 Java Stream API에서 터미널 동작 예시 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

    관련 라벨:
    원천:yisu.com
    본 웹사이트의 성명
    본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
    인기 튜토리얼
    더>
    최신 다운로드
    더>
    웹 효과
    웹사이트 소스 코드
    웹사이트 자료
    프론트엔드 템플릿