Stream流中间方法的使用

发布于:2025-06-30 ⋅ 阅读:(14) ⋅ 点赞:(0)

Stream流的中间方法详解

Stream流的中间方法指那些返回新Stream的操作,允许链式调用。这些方法通常用于数据过滤、映射、排序等操作,不会触发最终计算。

filter方法

filter用于筛选满足条件的元素,接受一个Predicate函数式接口参数。

List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5);
List<Integer> evenNumbers = numbers.stream()
    .filter(n -> n % 2 == 0)
    .collect(Collectors.toList()); 
// 输出: [2,4]

map方法

map用于元素转换,接受Function接口参数,将元素映射为另一种形式。

List<String> names = Arrays.asList("Alice", "Bob");
List<Integer> nameLengths = names.stream()
    .map(String::length)
    .collect(Collectors.toList());
// 输出: [5,3]

flatMap方法

flatMap用于将嵌套结构扁平化,每个元素转换为新的Stream后合并。

List<List<Integer>> nestedNumbers = Arrays.asList(
    Arrays.asList(1,2),
    Arrays.asList(3,4));
List<Integer> flattened = nestedNumbers.stream()
    .flatMap(List::stream)
    .collect(Collectors.toList());
// 输出: [1,2,3,4]

distinct方法

distinct去除重复元素,依赖元素的equals方法。

List<Integer> duplicates = Arrays.asList(1,2,2,3);
List<Integer> unique = duplicates.stream()
    .distinct()
    .collect(Collectors.toList());
// 输出: [1,2,3]

sorted方法

sorted用于排序,可自然排序或提供Comparator。

List<String> unsorted = Arrays.asList("Bob", "Alice");
List<String> sorted = unsorted.stream()
    .sorted()
    .collect(Collectors.toList());
// 输出: ["Alice","Bob"]

List<Integer> numbers = Arrays.asList(3,1,2);
List<Integer> customSorted = numbers.stream()
    .sorted(Comparator.reverseOrder())
    .collect(Collectors.toList());
// 输出: [3,2,1]

limit方法

limit截取前N个元素。

List<Integer> numbers = Arrays.asList(1,2,3,4,5);
List<Integer> firstThree = numbers.stream()
    .limit(3)
    .collect(Collectors.toList());
// 输出: [1,2,3]

skip方法

skip跳过前N个元素。

List<Integer> numbers = Arrays.asList(1,2,3,4,5);
List<Integer> afterTwo = numbers.stream()
    .skip(2)
    .collect(Collectors.toList());
// 输出: [3,4,5]

peek方法

peek用于调试,允许查看流经的元素但不修改。

List<String> names = Arrays.asList("Alice", "Bob");
List<String> result = names.stream()
    .peek(System.out::println)
    .map(String::toUpperCase)
    .collect(Collectors.toList());
// 输出调试信息: Alice Bob
// 最终结果: ["ALICE","BOB"]
实例一:
public class StreamDemo6 {
    public static void main(String[] args) {
        /*
            filter              过滤
            limit               获取前几个元素
            skip                跳过前几个元素

            注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程
            注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据
        */

        ArrayList<String> list = new ArrayList<>();
        Collections.addAll(list, "张无忌", "周芷若", "赵敏", "张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤");

        //filter   过滤  把张开头的留下,其余数据过滤不要
        /*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("张"))
                .filter(s -> s.length() == 3)
                .forEach(s -> System.out.println(s));


        System.out.println("====================================");
        System.out.println(list);*/


       /* limit               获取前几个元素
        skip                跳过前几个元素*/

        //"张无忌", "周芷若", "赵敏", "张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤"
        //list.stream().limit(3).forEach(s -> System.out.println(s));
        //list.stream().skip(4) .forEach(s -> System.out.println(s));


        //课堂练习:
        //"张强", "张三丰", "张翠山"
        //第一种思路:
        //先获取前面6个元素:"张无忌", "周芷若", "赵敏", "张强", "张三丰", "张翠山",
        //然后跳过前面3个元素
        //list.stream().limit(6).skip(3).forEach(s -> System.out.println(s));

        //第二种思路:
        //先跳过3个元素:"张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤"
        //然后再获取前面3个元素:"张强", "张三丰", "张翠山"
        //list.stream().skip(3).limit(3).forEach(s -> System.out.println(s));

    }
}
实例二:
public class StreamDemo7 {
    public static void main(String[] args) {
        /*
            distinct            元素去重,依赖(hashCode和equals方法)
            concat              合并a和b两个流为一个流

            注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程
            注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据
        */

        ArrayList<String> list1 = new ArrayList<>();
        Collections.addAll(list1, "张无忌","张无忌","张无忌", "张强", "张三丰", "张翠山", "张良", "王二麻子", "谢广坤");

        ArrayList<String> list2 = new ArrayList<>();
        Collections.addAll(list2, "周芷若", "赵敏");


        // distinct            元素去重,依赖(hashCode和equals方法)
        //list1.stream().distinct().forEach(s -> System.out.println(s));


        Stream.concat(list1.stream(),list2.stream()).forEach(s -> System.out.println(s));


    }
}
实例三:
public class StreamDemo8 {
    public static void main(String[] args) {
         /*
            map                 转换流中的数据类型

            注意1:中间方法,返回新的Stream流,原来的Stream流只能使用一次,建议使用链式编程
            注意2:修改Stream流中的数据,不会影响原来集合或者数组中的数据
        */

        ArrayList<String> list = new ArrayList<>();
        Collections.addAll(list, "张无忌-15", "周芷若-14", "赵敏-13", "张强-20", "张三丰-100", "张翠山-40", "张良-35", "王二麻子-37", "谢广坤-41");
        //需求:只获取里面的年龄并进行打印
        //String->int

        //第一个类型:流中原本的数据类型
        //第二个类型:要转成之后的类型

        //apply的形参s:依次表示流里面的每一个数据
        //返回值:表示转换之后的数据

        //当map方法执行完毕之后,流上的数据就变成了整数
        //所以在下面forEach当中,s依次表示流里面的每一个数据,这个数据现在就是整数了
        list.stream().map(new Function<String, Integer>() {
            @Override
            public Integer 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("------------------------");


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



    }
}

注意:中间操作具有延迟特性,只有遇到终端操作时才会执行。多个中间操作可以组合形成处理流水线。


网站公告

今日签到

点亮在社区的每一天
去签到