Java中的Stream流

一、介绍

1. Stream流的作用

结合了Lambda表达式,简化集合、数组的操作。

2. Stream流的使用步骤

①先得到一条Stream流,并把数据放上去;

|--------|-----------------------------------------------------|------------------|
| 获取方式 | 方法名 | 说明 |
| 单列集合 | default Stream<E> stream() | Collection中的默认方法 |
| 双列集合 | 无 | 无法直接使用stream流 |
| 数组 | public static <T> Stream<T> stream(T[] array) | Arrays工具类中的静态方法 |
| 一堆零散数据 | public static <T> Stream<T> of(T...values) | Stream接口中的静态方法 |

(1)单列集合:
java 复制代码
import java.util.ArrayList;
import java.util.Collections;
import java.util.function.Consumer;
import java.util.stream.Stream;

public class StreamDemo2 {
    public static void main(String[] args) {
        // 1. 单列集合获取Stream流
        ArrayList<String> list = new ArrayList<>();
        Collections.addAll(list, "a", "b", "c", "d", "e");
        /*// 获取一条流水线,并把集合中的数据放到流水线上
        Stream<String> stream1 = list.stream();
        // 使用终结方法打印一条流水线上的所有数据
        stream1.forEach(new Consumer<String>() {
            @Override
            public void accept(String s) {
                // s:依次表示流水线上的每一个数据
                System.out.println(s);
            }
        });*/

        // 链式编程
        list.stream().forEach(s -> System.out.println(s));
    }
}
(2)双列集合
java 复制代码
import java.util.HashMap;

public class StreamDemo3 {
    public static void main(String[] args) {
        // 双列集合
        // 1. 创建双列集合
        HashMap<String, Integer> hm = new HashMap<>();
        // 2. 添加数据
        hm.put("aaa", 111);
        hm.put("bbb", 222);
        hm.put("ccc", 333);
        hm.put("ddd", 444);
        hm.put("eee", 555);
        // 3. 第一种方式 获取stream流
        hm.keySet().stream().forEach(s -> System.out.println(s));
        // 4. 第二种方式 获取stream流
        hm.entrySet().stream().forEach(s -> System.out.println(s));

    }
}
(3)数组
java 复制代码
import java.util.Arrays;
import java.util.stream.Stream;

public class StreamDemo4 {
    public static void main(String[] args) {
        // 数组获取stream流

        // 1. 创建数组
        int[] arr1 = {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};

        String[] arr2 = {"a", "b", "c"};
        // 2. 获取stream流
        Arrays.stream(arr1).forEach(s -> System.out.println(s));

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

        Arrays.stream(arr2).forEach(s -> System.out.println(s));

        // 注意:
        // Stream接口中静态方法of的细节
        // 方法的形参是一个可变参数,可以传递一堆零散的数据,也可以传递数组
        // 但是数组必须是引用数据类型的,如果传递基本数据类型,是会把整个数组当作一个元素,放到stream当中。。
        Stream.of(arr1).forEach(s -> System.out.println(s));  // [I@214c265e
        Stream.of(arr2).forEach(s -> System.out.print(s + " "));  // a b c 
    }
}
(4)一堆零散数据
java 复制代码
import java.util.stream.Stream;

public class StreamDemo5 {
    public static void main(String[] args) {
        // 一堆零散数据获取stream流

        Stream.of(1, 2, 3, 4, 5).forEach(s -> System.out.println(s));

        Stream.of("a", "b", "c", "d", "e").forEach(s -> System.out.println(s));
    }
}

②利用中间方法对流水线上的数据进行各种操作;

注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程

注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据

|-------------------------------------------------------|----------------------------|
| 名称 | 说明 |
| Stream<T> filter(Predicate<? super<T> predicate) | 过滤 |
| Stream<T> limit(long maxSize) | 获取前几个元素 |
| Stream<T> skip(long n) | 跳过前几个元素 |
| Stream<T> distinct() | 元素去重,依赖(hashCode和equals方法) |
| static<T> Stream<T> concat(Stream a, Stream b) | 合并a和b两个流为一个流 |
| Stream<R> map(Function<T, R> mapper) | 转换流中的数据类型 |

java 复制代码
import java.util.ArrayList;
import java.util.Collections;
import java.util.stream.Stream;


public class StreamDemo6 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        Collections.addAll(list, "张无忌", "张无忌", "张无忌", "周芷若", "赵敏", "张强", "张三丰");

        // 1. fliter 过滤 把以"张"开头的留下,其余数据过滤不要
        /*list.stream().filter(new Predicate<String>() {
            @Override
            public boolean test(String s) {
                // 如果返回值为true,表示当前数据留下
                // 如果返回真为false,表示当前数据舍弃不要
                return s.startsWith("张");
            }
        }).forEach(s -> System.out.println(s));*/

        list.stream().filter(s -> s.startsWith("张")).forEach(s -> System.out.println(s));
        System.out.println("------------------------------");
        System.out.println(list);  // 注意:修改Stream流中的数据,不会影响原理集合或者数组中的数据
        System.out.println("------------------------------");

        /*Stream<String> stream1 = list.stream().filter(s -> s.startsWith("张"));
        Stream<String> stream2 = stream1.filter(s -> s.length() == 3);
        stream2.forEach(s -> System.out.println(s));
        Stream<String> stream3 = stream1.filter(s -> s.length() == 3);  // java.lang.IllegalStateException*/

        // 2. limit 获取前几个元素
        list.stream().limit(3).forEach(s -> System.out.println(s));
        System.out.println("------------------------------");

        // 3. skip 跳过前几个元素
        list.stream().skip(3).forEach(s -> System.out.println(s));
        System.out.println("------------------------------");

        // 4. distinct 元素去重
        list.stream().distinct().forEach(s -> System.out.println(s));
        System.out.println("------------------------------");

        // 5. concat 合并a和b两个流为一个流
        ArrayList<String> list2 = new ArrayList<>();
        Collections.addAll(list2, "周芷若", "赵六");
        Stream.concat(list.stream(), list2.stream()).forEach(s -> System.out.println(s));
        System.out.println("------------------------------");

        // 6. map 转换流中的数据类型
        ArrayList<String> list3 = new ArrayList<>();
        Collections.addAll(list3, "张无忌-15", "周芷若-14", "赵敏-13", "张强-20", "张三丰-100", "张翠山-40", "张良-35");
        // 只获取里面的年龄并进行打印
        // 当map方法执行完毕后,流上的数据就变成了整数
        // 所以再下面的foreach当中,s一次表示流里面的每一个数据,这个数据现在就是整数了
        /*list3.stream().map(new Function<String, Object>() {
            @Override
            public Object apply(String s) {
                String[] arr = s.split("-");
                String ageString = arr[1];
                int age = Integer.parseInt(ageString);
                return age;
            }
        }).forEach(s -> System.out.println(s));
        System.out.println("------------------------------");*/

        list3.stream()
                .map(s -> Integer.parseInt(s.split("-")[1]))
                .forEach(s -> System.out.println(s));
    }
}

③使用终结方法对流水线上的数据进行操作

|-------------------------------|-------------------|
| 名称 | 说明 |
| void forEach(Consumer action) | 遍历 |
| long count() | 统计 |
| toArray() | 收集流中的数据,放到数组中 |
| collect(Collecctor collector) | 收集流中的数据,放到集合中 |

java 复制代码
import java.util.*;
import java.util.function.Function;
import java.util.function.IntFunction;
import java.util.stream.Collectors;

public class StreamDemo7 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        Collections.addAll(list, "张无忌", "周芷若", "赵敏", "张强", "张三丰");

        // 1. forEach 遍历
        // Consumer的泛型:表示流中数据的类型
        // accept方法的形参s:依次表示流里面的每一个数据
        // 方法体:对每一个数据的处理操作(打印)
       /* list.stream().forEach(new Consumer<String>() {
            @Override
            public void accept(String s) {
                System.out.println(s);
            }
        });*/

        list.stream().forEach(s -> System.out.println(s));
        System.out.println("--------------------------");

        // 2. count 统计
        long count = list.stream().count();
        System.out.println(count);
        System.out.println("--------------------------");

        // 3. toArray 收集流中的数据放到数组中
        Object[] arr1 = list.stream().toArray();
        System.out.println(Arrays.toString(arr1));
        System.out.println("--------------------------");

        // IntFunction的泛型:具体类型的数组
        // apply的形参:流中的数据的个数,要跟数组的长度保持一致
        // apply方法的返回值:具体类型的数组
        // 方法体:就是创建数组

        // toArray方法的参数的作用:负责创建一个指定类型的数组
        // toArray方法的底层:会依次得到流里面的每一个数据,并把数据放到数组当中
        // toArray方法的返回值:是一个装着流里面所有数据的数组
       /* String[] arr2 = list.stream().toArray(new IntFunction<String[]>() {
            @Override
            public String[] apply(int value) {
                return new String[value];
            }
        });
        System.out.println(Arrays.toString(arr2));
        System.out.println("--------------------------");*/

        String[] arr3 = list.stream().toArray(value -> new String[value]);
        System.out.println(Arrays.toString(arr3));
        System.out.println("--------------------------");


        // 4. 收集到List集合当中
        // 需求:把所有的男性收集起来
        ArrayList<String> list2 = new ArrayList<>();
        Collections.addAll(list2, "张无忌-男-15", "周芷若-女-14", "赵敏-女-13", "张强-男-20", "张三丰-男-100", "张翠山-男-40", "张良-男-35");
        List<String> newList = list2.stream()
                .filter(s -> "男".equals(s.split("-")[1]))
                .collect(Collectors.toList());
        System.out.println(newList);
        System.out.println("--------------------------");

        // 4. 收集到Set集合当中
        // 需求:把所有的男性收集起来
        Set<String> newSet = list2.stream().filter(s -> "男".equals(s.split("-")[1]))
                .collect(Collectors.toSet());  // 会自动去重
        System.out.println(newSet);
        System.out.println("--------------------------");

        // 4. 收集到Map集合当中
        // 谁作为键,谁作为值。把所有的男性收集起来
        // 键:姓名 值:年龄
        // 注意点:如果我们要收集数据到Map当中,键是不能重复的
       /* Map<String, Integer> newMap = list2.stream()
                .filter(s -> "男".equals(s.split("-")[1]))
                *//*
                toMap:参数一:键的生成规则。参数二:值的生成规则
                参数一:Function泛型一:表示流中每一个数据的类型
                               泛型二:表示Map集合中键的数据类型
                       方法apply形参:依次表示流里面的每一个数据
                               方法体:生成键的代码
                               返回值:已经生成的键
                参数二:Function泛型一:表示流中每一个数据的类型
                               泛型二:表示Map集合中值的数据类型
                       方法apply形参:依次表示流里面的每一个数据
                               方法体:生成值的代码
                               返回值:已经生成的值
                 *//*
                .collect(Collectors.toMap(new Function<String, String>() {
                                              @Override
                                              public String apply(String s) {
                                                  return s.split("-")[0];  // 姓名
                                              }
                                          },
                        new Function<String, Integer>() {
                            @Override
                            public Integer apply(String s) {
                                return Integer.parseInt(s.split("-")[2]);
                            }
                        }));

        System.out.println(newMap);
        System.out.println("--------------------------");*/
        // lambda表达式
        Map<String, Integer> map2 = list2.stream()
                .filter(s -> "男".equals(s.split("-")[1]))
                .collect(Collectors.toMap(
                        s -> s.split("-")[0]
                        ,
                        s -> Integer.parseInt(s.split("-")[2])
                ));
        System.out.println(map2);
    }
}

二、案例分析

1. 要求:按照下面的要求完成集合的创建和遍历。

①把所有以"张"开头的元素存储到新集合中;

②把"张"开头的,长度为3的元素再存储到新集合中;

③遍历打印最终结果。

java 复制代码
import java.util.ArrayList;

public class StreamDemo1 {
    public static void main(String[] args) {
        ArrayList<String> list = new ArrayList<>();
        list.add("张无忌");
        list.add("周芷若");
        list.add("赵敏");
        list.add("张强");
        list.add("张三丰");

        list.stream().filter(name -> name.startsWith("张")).filter(name -> name.length() == 3).forEach(name -> System.out.println(name));
        // 张无忌
        // 张三丰
    }
}

2. 数据过滤

定义一个集合,并添加一些数据1,2,3,4,5,6,7,8,9,10。过滤奇数,只留下偶数,并将结果保存起来。

java 复制代码
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;

public class StreamDemo8 {
    public static void main(String[] args) {
        /*
        定义一个集合,并添加一些数据1,2,3,4,5,6,7,8,9,10。
        过滤奇数,只留下偶数,并将结果保存起来。
        */

        // 1. 定义一个集合
        ArrayList<Integer> list = new ArrayList<>();
        // 2. 添加一些数据
        Collections.addAll(list, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
        // 3. 过滤奇数,只留下偶数
        // 进行判断,如果是偶数,返回true保留
        List<Integer> newList = list.stream()
                .filter(n -> n % 2 == 0)
                .collect(Collectors.toList());
        // 4. 打印集合
        System.out.println(newList);
    }
}

3. 数据操作

创建一个ArrayList集合,并添加以下字符串,字符串中前面是姓名,后面是年龄。"zhangsan, 23", "lisi, 24", "wangwu, 25"。保留年龄大于等于24岁的人,并将结果收到到Map集合中,姓名为键,年龄为值。

java 复制代码
import java.util.ArrayList;
import java.util.Map;
import java.util.stream.Collectors;

public class StreamDemo9 {
    public static void main(String[] args) {
        /*
        创建一个ArrayList集合,并添加以下字符串,字符串中前面是姓名,
        后面是年龄。"zhangsan, 23", "lisi, 24", "wangwu, 25"。
        保留年龄大于等于24岁的人,并将结果收到到Map集合中,
        姓名为键,年龄为值。
         */

        // 1. 创建一个ArrayList集合
        ArrayList<String> list = new ArrayList<>();
        // 2. 添加以下字符串
        list.add("zhangsan, 23");
        list.add("lisi, 24");
        list.add("wangwu, 25");
        // 3. 保留年龄大于等于24岁的人
        Map<String, Integer> map = list.stream()
                .filter(s -> Integer.parseInt(s.split(", ")[1]) >= 24)
                .collect(Collectors.toMap(
                        s -> s.split(", ")[0],
                        s -> Integer.parseInt(s.split(", ")[1])));
        System.out.println(map);
    }
}

4. 数据操作

现有两个ArrayList集合,第一个集合:存储6名男演员的名字和年龄。第二个集合:存储6名女演员的名字和年龄。姓名和年龄中间用逗号隔开。比如,张三,23。

要求完成下面的操作:①男演员只要名字为3个字的前两个人;②女演员只要姓杨的,并且不要第一个;③把过滤后的男演员姓名和女演员姓名合并到一起;④将上一步的演员信息封装成Actor对象;⑤将所有的演员对象都保存到List集合中。

备注:演员类Actor,属性只有一个:name,age

java 复制代码
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.function.Function;
import java.util.stream.Collectors;
import java.util.stream.Stream;

public class StreamDemo10 {
    public static void main(String[] args) {
        /*
        现有两个ArrayList集合,第一个集合:存储6名男演员的名字和年龄。
        第二个集合:存储6名女演员的名字和年龄。姓名和年龄中间用逗号隔开。比如,张三,23。
        要求完成下面的操作:①男演员只要名字为3个字的前两个人;
        ②女演员只要姓杨的,并且不要第一个;
        ③把过滤后的男演员姓名和女演员姓名合并到一起;
        ④将上一步的演员信息封装成Actor对象;⑤将所有的演员对象都保存到List集合中。
        备注:演员类Actor,属性只有一个:name,age
         */

        // 1. 创建两个ArrayList集合
        ArrayList<String> manList = new ArrayList<>();
        ArrayList<String> womenList = new ArrayList<>();
        // 2. 添加数据
        Collections.addAll(manList, "小小蔡, 24", "小小叶, 23", "小小刘, 22", "小吴, 25", "小小谷, 23", "小李, 26");
        Collections.addAll(womenList, "杨小甜, 23", "杨小小, 21", "杨小鞠, 22", "小单, 24", "刘小陈, 22", "小小曲, 23");
        // 3. 男演员只要名字为3个字的前两个人
        Stream<String> stream1 = manList.stream()
                .filter(s -> s.split(", ")[0].length() == 3)
                .limit(2);

        // 4. 女演员只要姓杨的,并且不要第一个
        Stream<String> stream2 = womenList.stream()
                .filter(s -> s.split(", ")[0].startsWith("杨"))
                .skip(1);

        // 5. 把过滤后的男演员姓名和女演员姓名合并到一起
        // 6. 将上一步的演员信息封装成Actor对象(类型转换)
        /*Stream.concat(stream1, stream2).map(new Function<String, Actor>() {
            @Override
            public Actor apply(String s) {
                String name = s.split(", ")[0];
                int age = Integer.parseInt(s.split(", ")[1]);
                return new Actor(name, age);
            }
        }).forEach(s -> System.out.println(s));*/
        List<Actor> list = Stream.concat(stream1, stream2)
                .map(s -> new Actor(s.split(", ")[0], Integer.parseInt(s.split(", ")[1])))
                .collect(Collectors.toList());
        System.out.println(list);

    }
}
相关推荐
陈王卜5 分钟前
django+boostrap实现发布博客权限控制
java·前端·django
小码的头发丝、5 分钟前
Spring Boot 注解
java·spring boot
java亮小白199710 分钟前
Spring循环依赖如何解决的?
java·后端·spring
飞滕人生TYF16 分钟前
java Queue 详解
java·队列
武子康38 分钟前
大数据-230 离线数仓 - ODS层的构建 Hive处理 UDF 与 SerDe 处理 与 当前总结
java·大数据·数据仓库·hive·hadoop·sql·hdfs
武子康40 分钟前
大数据-231 离线数仓 - DWS 层、ADS 层的创建 Hive 执行脚本
java·大数据·数据仓库·hive·hadoop·mysql
苏-言1 小时前
Spring IOC实战指南:从零到一的构建过程
java·数据库·spring
界面开发小八哥1 小时前
更高效的Java 23开发,IntelliJ IDEA助力全面升级
java·开发语言·ide·intellij-idea·开发工具
草莓base1 小时前
【手写一个spring】spring源码的简单实现--容器启动
java·后端·spring
Allen Bright1 小时前
maven概述
java·maven