深入理解Java8新特性之Stream API的创建方式和中间操作步骤

目录
  • 1.什么是StreamAPI?
  • 2.Stream API操作的三个步骤
    • 2.1 创建Stream
    • 2.2 中间操作
      • 2.2.1 中间操作之筛选与切片
      • 2.2.2 中间操作之映射
      • 2.2.3 中间操作之排序

1.什么是StreamAPI?

Java8中有两大最为重要的改变。第一个是 Lambda 表达式;另外一个则是 m Stream API (java.util.stream.*) 。

Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找、过滤和映射数据等操作。

使用Stream API 对集合数据进行操作,就类似于使用 SQL 执行的数据库查询。也可以使用 Stream API 来并行执行操作。简而言之,Stream API 提供了一种高效且易于使用的处理数据的方式。

流 (Stream) 到底是什么 呢 ?

是数据渠道,用于操作数据源(集合、数组等)所生成的元素序列。“ 集合讲的是 数据 , 流讲的是 计算 !”

注意 :

  • Stream 自己不会存储元素。
  • Stream 不会改变源对象。相反,他们会返回一个持有结果的新Stream。
  • Stream 操作是延迟执行的。这意味着他们会等到需要结果的时候才执行。

2.Stream API操作的三个步骤

  • 创建Stream:一个数据源(如:集合、数组),获取一个流。
  • 中间操作:一个中间操作链,对数据源的数据进行处理。
  • 终止操作( ( 终端操作) ):一个终止操作,执行中间操作链,并产生结果。

2.1 创建Stream

在Java8 中, Collection 接口被扩展,提供了两个获取流的方法 :

  • default Stream<E> stream() : 返回一个顺序流
  • default Stream<E> parallelStream() : 返回一个并行流

Java8 中的 Arrays 的静态方法 stream() 可以获取数组流:

  • static <T> Stream<T> stream(T[] array): 返回一个流重载形式 , 能够处理对应基本类型的数组 :
  • public static IntStream stream(int[] array)
  • public static LongStream stream(long[] array)
  • public static DoubleStream stream(double[] array)

可以使用静态方法 Stream.of(),通过显示值创建一个流。它可以接收任意数量的参数。

  • public static<T> Stream<T> of(T... values) : 返回一个流

可以使用静态方法 Stream.iterate() 和 Stream.generate(),创建无限流。

  • 迭代 public static<T> Stream<T> iterate(final T seed, final UnaryOperator<T> f)
  • 生成 public static<T> Stream<T> generate(Supplier<T> s)
    //创建Stream
    @Test
    public void test1() {
        //1.Collection 提供了两个方法  stream() 与 parallelStream()
        List<String> list = new ArrayList<>();
        Stream<String> stream1 = list.stream();

        //2. 通过 Arrays 中的 stream() 获取一个数组流
        String[] strings = new String[10];
        Stream<String> stream2 = Arrays.stream(strings);

        //3. 通过 Stream 类中静态方法 of()
        Stream<String> stream3 = Stream.of("abc","666","???");

        //4. 创建无限流
        //迭代
        Stream<Integer> stream4 = Stream.iterate(0, (x) -> x + 2);
        stream4.limit(5).forEach(System.out::println);

        //生成
        Stream.generate(() -> Math.random())
                .limit(5)
                .forEach(System.out::println);
    }

2.2 中间操作

多个 中间操作可以连接起来形成一个 流 水 线,除非流水线上触发终止操作,否则 中 间操作 不 会执行 任 何 的 处 理!

而在 终止操作时一次性全部处理 , 称 为 “ 惰 性 求 值 ”。 常用的中间操作大体上可以分为三类:①筛选与切片;②映射;③排序

2.2.1 中间操作之筛选与切片

下面通过一些代码案例来应用一下。 首先,我们准备一个Employee类,以及一个List集合。

package com.szh.java8;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

/**
 *
 */
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Employee {

    private Integer id;
    private String name;
    private Integer age;
    private Double salary;

    public Employee(Integer id) {
        this.id = id;
    }

    public Employee(Integer id,String name) {
        this.id = id;
        this.name = name;
    }

}
List<Employee> employees = Arrays.asList(
        new Employee(1001,"张三",26,6666.66),
        new Employee(1002,"李四",50,1111.11),
        new Employee(1003,"王五",18,9999.99),
        new Employee(1004,"赵六",35,8888.88),
        new Employee(1005,"田七",44,3333.33),
        new Employee(1005,"田七",44,3333.33),
        new Employee(1005,"田七",44,3333.33)
);

筛选年龄小于35岁的员工信息。 这里每次都是先 println 打印结果,然后进行比较,满足则输出。

    //内部迭代,在 Stream API 内部完成
    @Test
    public void test2() {
        //所有的中间操作不会做任何的处理
        Stream<Employee> stream = employees.stream()
                                    .filter((e) -> {
                                        System.out.println("测试中间操作");
                                        return e.getAge() <= 35;
                                    });
        //只有当做终止操作时,所有的中间操作会一次性的全部执行,称为“惰性求值”
        stream.forEach(System.out::println);
    }

筛选薪资大于5000的员工信息,但是只保留前两条。

    @Test
    public void test3() {
        employees.stream()
                .filter((e) -> {
                    System.out.println("短路....");
                    return e.getSalary() > 5000;
                })
                .limit(2)
                .forEach(System.out::println);
    }

筛选薪资大于5000的员工信息,但是扔掉前两条。

    @Test
    public void test4() {
        employees.stream()
                .filter((e) -> e.getSalary() > 5000)
                .skip(2)
                .forEach(System.out::println);
    }

对集合中的所有员工信息做去重处理。

    @Test
    public void test5() {
        employees.stream()
                .distinct()
                .forEach(System.out::println);
    }

2.2.2 中间操作之映射

这里仍然是借助上面案例中的Employee类、List集合完成。

首先对自定义的List集合中的所有字符串进行大写转换,同时对员工集合进行 name 字段的映射处理。

    @Test
    public void test1() {
        List<String> list = Arrays.asList("aaa","bbb","ccc","ddd","eee");
        list.stream()
            .map((str) -> str.toUpperCase())
            .forEach(System.out::println);

        System.out.println("--------------------------------------");

        employees.stream()
                .map(Employee::getName)
                .forEach(System.out::println);
    }

对List集合中的每个字符串中的每个字符进行遍历输出。(①使用map;②使用flatMap)

这两种方式主要区别就在获取到Stream流之后,流中的泛型不一样了。

我们使用map了话,它会将strList集合中元素以 "aaa"、"bbb"、"ccc" 这样的形式返回,此时Stream流的泛型就仍然是一个Stream流(在这个Stream流中才是我们想要的数据),也就是我们第一次对stream流进行遍历之后,拿到的是一个又一个的Character字符数组,然后还需要再对这个字符数组进行二次遍历(也就是代码上半部分中那两个 forEach)。

如果我们使用flatMap了话,它会将strList集合中元素以 "a'、"b"、"c"、"d" 这样的形式返回,所以此时Stream流的泛型就只是一个Character字符数组,那么一次遍历就可以取出数据了(也就是代码下半部分的那个forEach)。

    @Test
    public void test2() {
        List<String> strList = Arrays.asList("aaa","bbb","ccc","ddd","eee");

        Stream<Stream<Character>> stream1 = strList.stream()
                .map(MyTest6::filterCharacter);
        stream1.forEach((sm) -> {
            sm.forEach(System.out::println);
        });

        System.out.println("----------------------------------------");

        Stream<Character> stream2 = strList.stream()
                .flatMap(MyTest6::filterCharacter);
        stream2.forEach(System.out::println);
    }

    private static Stream<Character> filterCharacter(String str) {
        List<Character> list = new ArrayList<>();

        for (Character character : str.toCharArray()) {
            list.add(character);
        }

        return list.stream();
    }

2.2.3 中间操作之排序

对List集合中的字符串进行自然排序。对员工集合中的员工信息进行定制化排序(先按照年龄排序,年龄一样再按照姓名排序)。

    @Test
    public void test3() {
        List<String> strList = Arrays.asList("ccc","aaa","bbb","eee","ddd");
        strList.stream()
                .sorted()
                .forEach(System.out::println);

        System.out.println("----------------------------------------");

        employees.stream()
                .sorted((e1,e2) -> {
                    if (e1.getAge().equals(e2.getAge())) {
                        return e1.getName().compareTo(e2.getName());
                    } else {
                        return e1.getAge().compareTo(e2.getAge());
                    }
                })
                .forEach(System.out::println);
    }

以上就是深入理解Java8新特性之Stream API的创建方式和中间操作步骤的详细内容,更多关于Java Stream API的资料请关注我们其它相关文章!

(0)

相关推荐

  • Java8 Stream API 详细使用方法与操作技巧指南

    本文实例讲述了Java8 Stream API 详细使用方法与操作技巧.分享给大家供大家参考,具体如下: 1. 概述 Java 8 引入的一个重要的特性无疑是 Stream API.Stream 翻译过来是"流",突然想到的是大数据处理有个流式计算的概念,数据通过管道经过一个个处理器(Handler)进行筛选,聚合,而且流都具有向量性,强调的是对数据的计算处理,而集合强调的是数据集.Stream可以看做是一个可操作的数据集序列,它可以指定你希望对集合进行的操作,可以执行非常复杂的查找.

  • 深入理解Java8新特性之Stream API的终止操作步骤

    目录 1.写在前面 2.终止操作 2.1 终止操作之查找与匹配 2.2 终止操作之归约与收集 1.写在前面 承接了上一篇文章(说完了Stream API的创建方式及中间操作):深入理解Java8新特性之Stream API的创建方式和中间操作步骤. 我们都知道Stream API完成的操作是需要三步的:创建Stream → 中间操作 → 终止操作.那么这篇文章就来说一下终止操作. 2.终止操作 终端操作会从流的流水线生成结果.其结果可以是任何不是流的值,例如:List.Integer,甚至是 v

  • Java8新特性之Stream API详解

    一.前言 StreamAPI在Java8版本中使用,关注的是对数据的筛选.查找.存储等 它可以做的事情有:过滤.排序.映射.归约 二.使用流程 Stream实例化中间操作(过滤.排序.映射.规约)终止操作(匹配查找.归约.收集) 三.案例演示 public class EmployeeData { public static List<Employee> getEmployees(){ List<Employee> list = new ArrayList<>(); l

  • 一文带你掌握Java8强大的StreamAPI

    目录 Stream 概述 Stream 实例化 1.方式一:通过集合 2.方式二:通过数组 3.方式三:通过Stream的of() 4.方式四:创建无限流 Stream 中间操作 1.筛选与切片 2.映射 3.排序 Stream 终止操作 1.匹配与查找 2.归约 3.收集 Stream 概述 Stream API ( java.util.stream) 把真正的函数式编程风格引入到Java中.这是目前为止对Java类库最好的补充,因为Stream API可以极大提供Java程序员的生产力,让程

  • Java8中Lambda表达式使用和Stream API详解

    前言 Java8 的新特性:Lambda表达式.强大的 Stream API.全新时间日期 API.ConcurrentHashMap.MetaSpace.总得来说,Java8 的新特性使 Java 的运行速度更快.代码更少.便于并行.最大化减少空指针异常. 0x00. 前置数据 private List<People> peoples = null; @BeforeEach void before () { peoples = new ArrayList<>(); peoples

  • java8 Stream API之reduce使用说明

    本篇我们只讲reduce. reduce的作用是把stream中的元素给组合起来. 至于怎么组合起来:它需要我们首先提供一个起始种子,然后依照某种运算规则使其与stream的第一个元素发生关系产生一个新的种子,这个新的种子再紧接着与stream的第二个元素发生关系产生又一个新的种子,就这样依次递归执行,最后产生的结果就是reduce的最终产出,这就是reduce的算法最通俗的描述: 那么结合实际的业务场景来说,运用reduce我们可以做sum,min,max,average,所以这些我们称之为针

  • 熟练掌握Java8新特性之Stream API的全面应用

    1.写在前面 关于Stream API的内容,已经基本上说完了.大家可以参考我的这两篇文章: 深入理解Java8新特性之Stream API的创建方式和中间操作步骤.深入理解Java8新特性之Stream API的终止操作步骤 那么这篇文章主要就是说一个Stream API的简单应用练习题. 2.应用举例 首先,我们需要创建一个交易员类.交易类.存储这两个类对象的List集合. package com.szh.java8.exer; import lombok.AllArgsConstructo

  • java8使用Stream API方法总结

    Stream是java8中处理集合的关键抽象概念,它可以指定您希望对集合进行的操作,可以执行非常复杂的查找.过滤和映射数据等操作.使用Stream API对集合数据进行操作,就类似于使用SQL执行的数据库查询. Stream 的三个操作步骤 1.创建Stream. 得到Stream流的第一种方式: 可以通过Collection系列集合提供提供的Stream()或parallelStream @Test public void test1() { //可以通过Collection系列集合提供提供的

  • 深入理解Java8新特性之Stream API的创建方式和中间操作步骤

    目录 1.什么是StreamAPI? 2.Stream API操作的三个步骤 2.1 创建Stream 2.2 中间操作 2.2.1 中间操作之筛选与切片 2.2.2 中间操作之映射 2.2.3 中间操作之排序 1.什么是StreamAPI? Java8中有两大最为重要的改变.第一个是 Lambda 表达式:另外一个则是 m Stream API (java.util.stream.*) . Stream 是 Java8 中处理集合的关键抽象概念,它可以指定你希望对集合进行的操作,可以执行非常复

  • 深入理解Java8新特性之新日期时间API的应用

    目录 1.新旧对比(线程安全问题) 2.LocalDate 3.LocalTime 4.LocalDateTime 5.Instant 6.Duration.Period 7.TestTemporalAdjuster.TestTemporalAdjusters 8.DateTimeFormatter 1.新旧对比(线程安全问题) 我们先来看下面的代码:

  • 深入理解Java8新特性之Lambda表达式的基本语法和自定义函数式接口

    1.写在前面 目前我们学习Java主要用到的应该就是Java8了,或者说大部分企业当前使用的也是Java8.那么既然Java8的应用如此之广泛,一定有一些亮点所在: Lambda 表达式 函数式接口 方法引用与构造器引用 Stream API 接口中的默认方法与静态方法 新时间日期API 其他新特性 速度更快.代码更少(增加了新的语法 Lambda 表达式).强大的 Stream API.便于并行.最大化减少空指针异常 Optional. 2.为什么要使用Lambda表达式? Lambda 是一

  • java8新特性之stream的collect实战教程

    1.list转换成list 不带return方式 List<Long> ids=wrongTmpList.stream().map(c->c.getId()).collect(Collectors.toList()); 带return方式 // spu集合转化成spubo集合//java8的新特性 List<SpuBo> spuBos=spuList.stream().map(spu -> { SpuBo spuBo = new SpuBo(); BeanUtils.c

  • 详解Java8 新特性之日期API

    Java 8 在包java.time下包含了一组全新的时间日期API.下面的例子展示了这组新API里最重要的一些部分: 1.Clock 时钟 Clock类提供了访问当前日期和时间的方法,Clock是时区敏感的,可以用来取代 System.currentTimeMillis() 来获取当前的微秒数.某一个特定的时间点也可以使用Instant类来表示,Instant类也可以用来创建老的java.util.Date对象. Clock clock = Clock.systemDefaultZone();

  • java8新特性之stream流中reduce()求和知识总结

    1.stream().reduce()单字段求和 (1)普通数字求和 public static void test2(){ List<Integer> list= Arrays.asList(new Integer[]{1,2,3,4,5,6,7,8,9}); Integer sum=list.stream().reduce((x,y)->x+y).get(); System.out.println(sum); } 2.BigDecimal求和 public static void m

  • 深入理解Java8新特性之Optional容器类的应用

    1.Optional容器类 Optional<T> 类(java.util.Optional) 是一个容器类,代表一个值存在或不存在,原来用 null 表示一个值不存在,现在 Optional 可以更好的表达这个概念.并且可以避免空指针异常. 常用方法 : Optional.of(T t) : 创建一个 Optional 实例 Optional.empty() : 创建一个空的 Optional 实例 Optional.ofNullable(T t) : 若 t 不为 null,创建 Opti

随机推荐