文章目录

  • 一、流简介
  • 二、创建流
  • 2.1 由值创建流:of
  • 2.2 由列表创建流:stream
  • 2.3 由 Builder 创建流:build
  • 2.4 由文件生成流:lines
  • 2.5 由函数生成流
  • 2.5.1 迭代(如果不做限制,就是创建无限流):iterate
  • 2.5.2 生成:generate
  • 三、常用操作
  • 3.1 筛选
  • 3.1.1 filter
  • 3.1.2 limit
  • 3.2 映射
  • 3.2.1 map
  • 3.3 查找和匹配
  • 3.3.1 anyMatch
  • 3.3.2 allMatch
  • 3.3.3 findAny
  • 3.3.4 findFirst
  • 3.4 规约
  • 3.4.1 reduce
  • 3.4.1.1 元素求和
  • 3.4.1.2 最大值
  • 3.4.1.3 最小值
  • 3.5 数值流
  • 3.5.1 对象流映射到数值流
  • 3.5.1.1 mapToInt
  • 3.5.1.2 mapToDouble
  • 3.5.1.3 mapToLong
  • 3.5.2 数值流转换回对象流
  • 3.5.2.1 boxed
  • 3.5.3 数值流的应用



一、流简介

  • 流式 API 从 Java8 开始引入,支持链式书写。
  • 流在管道中流通,在节点被处理。
  • Stream是独特的,既不同于io,也不同于List。
  • Stream在理论上能容纳无限对象,List不能
  • 流不是一种数据结构,流存储的只是一种数据视图。
  • 流的计算是惰性计算,真正的计算通常只发生在最后的结果获取时。流式计算存在第一步、中间一步、最后一步的说法,只有当到达最后一步执行函数的时候,整个惰性函数才会执行。
  • 流只能消费一次,不能被两次消费

如下的代码,对stream进行了两次遍历,所以会报错

List<String> stringList = Arrays.asList("Java", "Python", "C++", "Matlab");
        Stream<String> stream = stringList.stream();
        stream.forEach(System.out::println);
        stream.forEach(System.out::println);

grpc java 流式 java流式编程原理_流式编程


二、创建流

2.1 由值创建流:of

方法一:使用静态方法 Stream.of() ,通过显式值创建一个流

Stream<String> stream = Stream.of("Java","Python","C++","Matlab");
String[] arr = new String[]{"Java","Python","C++","Matlab"};
        Stream<String> stream = Stream.of(arr);

2.2 由列表创建流:stream

方法二:使用 Collection 的 stream() 方法,创建一个流

List<String> list = Arrays.asList("Java","Python","C++","Matlab");
        Stream<String> stream = list.stream();

2.3 由 Builder 创建流:build

方法三:使用 builder() 创建一个流

Stream<String> stream = Stream.<String>builder()
                .add("Java")
                .add("Python")
                .add("C++")
                .add("Matlab")
                .build();

2.4 由文件生成流:lines

方法四:使用 Files.lines() 创建一个流

Stream<String> stream = null;
        try {
            stream = Files.lines(Paths.get("data.txt"), Charset.defaultCharset());
        }catch (IOException e){
            e.printStackTrace();
        }

2.5 由函数生成流

2.5.1 迭代(如果不做限制,就是创建无限流):iterate

Stream<Integer> stream = Stream.iterate(0, n -> n + 2)
                .limit(10);
        stream.forEach(System.out::println); // 0,2,4,6,8,10,12,14,16,18

下面用迭代的方法生成斐波那契数列:

Stream.iterate(new int[]{0, 1}, t -> new int[]{t[1], t[0] + t[1]})
                .limit(10)
                .forEach(t -> {
                    System.out.println(t[0] + " " + t[1]);
                });

输出如下:

0 1
1 1
1 2
2 3
3 5
5 8
8 13
13 21
21 34
34 55

2.5.2 生成:generate

下面是生成 4 个随机数的代码

// 0.24958123386777675,0.3934127178560789,0.45154850021573034,0.005268717434774417,
        Stream.generate(Math::random)
                .limit(4)
                .forEach(num -> System.out.print(num + ","));

三、常用操作

这是下面例子中会用到的学生对象,下面会利用流式编程,对学生列表进行一系列操作

static class Student{
        String name;
        int age;
        public Student(String name, int age) {
            this.name = name;
            this.age = age;
        }
        @Override
        public String toString() {
            return "Student{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    '}';
        }
    }

下面是获取学生列表的函数,后面都会以这个学生列表进行举例说明

public static List<Student> createStudentList(){
        return Arrays.asList(
                new Student("WSKH",21),
                new Student("WRQ",22),
                new Student("SXC",20),
                new Student("XCY",36),
                new Student("CZG",40),
                new Student("LHH",18)
        );
    }

在后面,假设我们已经通过上面的函数获取到了学生列表,并将其命名为 studentList

List<Student> studentList = createStudentList();

3.1 筛选

3.1.1 filter

filter():返回一个包括所有符合谓词的元素的流

应用实例:获取年龄在 25 以下的学生

List<Student> result = studentList.stream().filter(student -> {
            return student.age < 25;
        }).collect(Collectors.toList());

3.1.2 limit

limit(n):该方法会返回一个不超过给定长度的流

应用实例:获取年龄在25以下的学生,并且要求结果中的学生不超过2位

List<Student> result = studentList.stream().filter(student -> {
            return student.age < 25;
        }).limit(2).collect(Collectors.toList());

3.2 映射

3.2.1 map

map():对流中每一个元素应用函数

应用案例:获取所有学生的年龄

List<Integer> result = studentList.stream().map(student -> {
            return student.age;
        }).collect(Collectors.toList());

应用案例:获取所有学生的名字

List<String> result = studentList.stream().map(student -> {
            return student.name;
        }).collect(Collectors.toList());

3.3 查找和匹配

3.3.1 anyMatch

anyMatch():判断流中是否有一个元素能匹配给定的谓词

应用案例:判断是否有小于等于18岁的学生

boolean b = studentList.stream().anyMatch(student -> student.age <= 18);

3.3.2 allMatch

allMatch():判断流中的元素是否都能匹配给定的谓词

应用案例:判断是否所有的学生都大于18岁

boolean b = studentList.stream().allMatch(student -> student.age > 18);

3.3.3 findAny

findAny():返回当前流中的任意元素

应用案例:获取任意一个大于25岁的学生

Optional<Student> any = studentList.stream().filter(student -> {
            return student.age > 25;
        }).findAny();
        System.out.println(any); // Optional[Student{name='XCY', age=36}]

3.3.4 findFirst

findFirst():返回当前流中的第一个元素

应用案例:获取学生列表中第一个大于25岁的学生

Optional<Student> first = studentList.stream().filter(student -> {
            return student.age > 25;
        }).findFirst();
        System.out.println(first); // Optional[Student{name='XCY', age=36}]

3.4 规约

3.4.1 reduce

reduce(p1,p2):操作组合流中的所有元素以产生单个值。
reduce的第一个参数:初始值,相当于给求和或者求最大最小值一个初值
reduce的第二个参数:一个BinaryOperator来将两个元素结合起来产生一个新值

3.4.1.1 元素求和

应用案例:求出学生们的年龄总和

int sum = studentList.stream().map(student -> {
            return student.age;
        }).reduce(0, (a, b) -> a + b);
        System.out.println(sum); // 157

也可以简写如下,用 Integer::sum 简化 (a, b) -> a + b

int sum = studentList.stream().map(student -> {
            return student.age;
        }).reduce(0, Integer::sum);
        System.out.println(sum); // 157

3.4.1.2 最大值

应用案例:求出学生们的最大年龄

int maxAge = studentList.stream().map(student -> {
            return student.age;
        }).reduce(0, (a, b) -> a > b ? a : b);
        System.out.println(maxAge); // 40

3.4.1.3 最小值

应用案例:求出学生们的最小年龄

int minAge = studentList.stream().map(student -> {
            return student.age;
        }).reduce(Integer.MAX_VALUE, (a, b) -> a < b ? a : b);
        System.out.println(minAge); // 18

3.5 数值流

在Java中,因为Java泛型不支持基本类型,所以我们无法使用像Stream这样的形式来保存int,只能采用形如Integer这样的形式。但是频繁装箱、拆箱操作会牺牲编译器的大量性能。

所以为了提高效率,Java标准库提供了三种使用基本类型的Stream,它们的使用和标准的Stream没有太大区别,直接使用:

  • IntStream
  • LongStream
  • DoubleStream

他们也被称为数值流。

3.5.1 对象流映射到数值流

3.5.1.1 mapToInt

mapToInt():返回一个 IntStream(而不是一个Stream)

IntStream intStream = studentList.stream().mapToInt(student -> student.age);

3.5.1.2 mapToDouble

mapToDouble():返回一个 DoubleStream(而不是一个Stream)

DoubleStream doubleStream = studentList.stream().mapToDouble(student -> student.age);

3.5.1.3 mapToLong

mapToLong():返回一个 LongStream(而不是一个Stream)

LongStream longStream = studentList.stream().mapToLong(student -> student.age);

3.5.2 数值流转换回对象流

3.5.2.1 boxed

boxed():要把原始流转换成一般流(这里每个int都会装箱成一个Integer)

IntStream intStream = studentList.stream().mapToInt(student -> student.age);
        Stream<Integer> boxed = intStream.boxed();

3.5.3 数值流的应用

应用案例:求出学生们的年龄总和

int sum = studentList.stream().mapToInt(student -> student.age).sum();
        System.out.println(sum); // 157

应用案例:求出学生们的最大年龄

OptionalInt max = studentList.stream().mapToInt(student -> student.age).max();
        System.out.println(max.isPresent() ? max.getAsInt() : "不存在最大值"); // 40

应用案例:求出学生们的最小年龄

OptionalInt min = studentList.stream().mapToInt(student -> student.age).min();
        System.out.println(min.isPresent() ? min.getAsInt() : "不存在最小值"); // 18

应用案例:求出学生们的平均年龄

OptionalDouble average = studentList.stream().mapToInt(student -> student.age).average();
        System.out.println(average.isPresent() ? average.getAsDouble() : "不存在平均值"); // 26.166666666666668

应用案例:求出大于25岁的学生的数量

long count = studentList.stream().mapToInt(student -> student.age).filter(age -> {
            return age > 25;
        }).count();
        System.out.println(count); // 2