首页 > 编程语言 >29_Java中的Stream流

29_Java中的Stream流

时间:2022-12-26 12:11:36浏览次数:38  
标签:Java Stream stream list 29 System add println

Java中的Stream流

总的来所就是将一组数据转换为流进行操作过滤,然后再将其还原进行操作

一、体验Stream流

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

​ 创建一个集合,存储多个字符串元素

​ 把集合中所有以“张”开头的元素存储到一个新的集合

​ 把“张”开头的集合中的长度为3的元素存储到一个新的集合

​ 遍历上一步得到的集合

使用Stream流的方式完成过滤操作

​ list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length() == 3).forEach(System.out::println);

​ 直接阅读代码的字面意思即可完美展示无关逻辑方式的语义:生成流、过滤姓张,过滤长度为3、遍历打印

参考代码:

package com.itheima_01;

import java.util.ArrayList;

/*
    需求:按照下面的要求完成集合的创建和遍历
		创建一个集合,存储多个字符串元素
		把集合中所有以“张”开头的元素存储到一个新的集合
		把“张”开头的集合中的长度为3的元素存储到一个新的集合
		遍历上一步得到的集合
 */
public class StreamDemo {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("林青霞");
        list.add("张曼玉");
        list.add("王祖贤");
        list.add("柳岩");
        list.add("张敏");
        list.add("张无忌");

        //把集合中所有以“张”开头的元素存储到一个新的集合
        ArrayList<String> zhangList = new ArrayList<String>();

        for(String s : list){
            if(s.startsWith("张")){
                zhangList.add(s);
            }
        }

//        System.out.println(zhangList);

        //把“张”开头的集合中的长度为3的元素存储到一个新的集合
        ArrayList<String> threeList = new ArrayList<String>();

        for(String s : zhangList){
            if(s.length() == 3){
                threeList.add(s);
            }
        }

//        System.out.println(threeList);

        for(String s : threeList){
            System.out.println(s);
        }

        //可以看到这样一个简单的过滤过程,就用了这么麻烦的步骤

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

        //使用Stream流来改进
//        list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length() == 3).forEach(s -> System.out.println(s));
        list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length() == 3).forEach(System.out::println);
    }
}

Stream流的使用

​ 生成流

​ 通过数据源(集合,数组等)生成流

​ list.stream()

​ 中间操作

​ 一个流后面可以跟随零个或者多个中间操作,其主要目的是打开流,做出某种程度的数据过滤/映射,然后返回一个新的流,交 给下一个操作使用

​ filter()

​ 终结操作

​ 一个流只能有一个终结操作,当这个操作执行后,流就被使用“光”了,无法在被操作。所以这必定是最后一个操作

​ forEach()

二、Stream流的生成方式

Stream流的常见生成方式:

​ Collection体系的集合可以使用默认方法stream()生成流

​ default Stream stream()

​ Map体系的集合间接生成流

​ 数组可以通过Stream接口的静态方法of(T...values)生成流

参考代码:

package com.itheima_02;

import java.util.*;
import java.util.stream.Stream;

/*
    Stream流的常见生成方式:
​		Collection体系的集合可以使用默认方法stream()生成流
​				default Stream<E> stream()
​		Map体系的集合间接生成流
​		数组可以通过Stream接口的静态方法of(T...values)生成流
 */
public class StreamDemo {
    public static void main(String[] args){
//        Collection体系的集合可以使用默认方法stream()生成流
        List<String> list = new ArrayList<String>();
        Stream<String> listStream = list.stream();

        Set<String> set = new HashSet<String>();
        Stream<String> setStream = set.stream();

//        Map体系的集合间接生成流
        Map<String, Integer> map = new HashMap<String, Integer>();
        Stream<String> keyStream = map.keySet().stream();   //键所对应的流
        Stream<Integer> valueStream =  map.values().stream();   //值所对应的流
        Stream<Map.Entry<String,Integer>> entryStream = map.entrySet().stream();    //键值对所对应的流



//        数组可以通过Stream接口的静态方法of(T...values)生成流
        String[] strArray = {"hello", "world","java"};
        Stream<String> strArrayStream = Stream.of(strArray);
        //通过可变参数的方式来转换
        Stream<String> strArrayStream2 = Stream.of("hello","world","java");
        Stream<Integer> integerStream = Stream.of(10, 20, 30);
    }
}

三、Stream流的常见中间操作方法

Stream filter(Predicate predicate):用于对流中的数据进行过滤

​ Predicate接口中的方法 boolean test(T t):对给定的参数进行判断,返回一个boolean值

参考代码:

package com.itheima_03;

import java.util.ArrayList;
import java.util.stream.Stream;

/*
Stream<T> filter(Predicate predicate):用于对流中的数据进行过滤

		Predicate接口中的方法			boolean test(T t):对给定的参数进行判断,返回一个boolean值
 */
public class StreamDemo01 {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("林青霞");
        list.add("张曼玉");
        list.add("王祖贤");
        list.add("柳岩");
        list.add("张敏");
        list.add("张无忌");

        //需求1:把list集合中以张开头的元素在控制台输出
        Stream<String> listStream = list.stream();
        listStream.filter(s -> s.startsWith("张")).forEach(s -> System.out.println(s));

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

        //需求2:把list集合中长度为3的元素在控制台输出
        //此处运行会报错:因为这个是创建好的流对象,已经在上面执行过一次操作了,而流对象只能执行一次操作
//        listStream.filter(s -> s.length() == 3).forEach(s -> System.out.println(s));
        list.stream().filter(s -> s.length() == 3).forEach(s -> System.out.println(s));

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

        //需求3:把list集合中以张开头的,长度为3的元素在控制台输出
        list.stream().filter(s -> s.startsWith("张")).filter(s -> s.length() == 3).forEach(System.out::println);
    }
}

Stream limit(long maxSize):返回此流中的元素组成的流,根据参数截取前几个元素(闭区间)

Stream skip(long n):跳过指定参数个数的数据,返回由该流的剩余元素组成的流(闭区间)

参考代码:

package com.itheima_03;

import java.util.ArrayList;

/*
Stream<T> limit(long maxSize):返回此流中的元素组成的流,根据参数截取前几个元素

Stream<T> skip(long n):跳过指定参数个数的数据,返回由该流的剩余元素组成的流
 */
public class StreamDemo02 {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("林青霞");
        list.add("张曼玉");
        list.add("王祖贤");
        list.add("柳岩");
        list.add("张敏");
        list.add("张无忌");

        //需求1:取前3个数据在控制台输出
        list.stream().limit(3).forEach(System.out::println);

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

        //需求2:跳过3个元素,把剩下的元素在控制台输出
        list.stream().skip(3).forEach(System.out::println);

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

        //需求3:跳过2个元素,把剩下的元素中前2个在控制台输出
        list.stream().skip(2).limit(2).forEach(System.out::println);
    }
}

static Stream concat(Stream a, Stream b):合并a和b两个流为一个流

Stream distinct():返回由该流的不同元素(根据Objectequals(Object))组成的流,即去掉重复元素

参考代码:

package com.itheima_03;

import java.util.ArrayList;
import java.util.stream.Stream;

/*
static<T> Stream<T> concat(Stream a, Stream b):合并a和b两个流为一个流

Stream<T> distinct():返回由该流的不同元素(根据Objectequals(Object))组成的流
 */
public class StreamDemo03 {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("林青霞");
        list.add("张曼玉");
        list.add("王祖贤");
        list.add("柳岩");
        list.add("张敏");
        list.add("张无忌");

        //需求1:取前4个数组组成一个流
        Stream<String> s1 = list.stream().limit(4);

        //需求2:跳过2个数据组成一个流
        Stream<String> s2 = list.stream().skip(2);

        //需求3:合并需求1和需求2得到的流,并把结果在控制台输出
//        Stream.concat(s1, s2).forEach(System.out::println); //将前四个和后四个合并

        //需求4:合并需求1和需求2得到的流,并把结果在控制台输出,要求字符串元素不能重复
        //去重后再输出
        Stream.concat(s1, s2).distinct().forEach(System.out::println);
    }
}

Stream sorted():返回由此流的元素组成的流,根据自然排序

Stream sorted(Comparator comparator):返回由该流的元素组成的流,根据提供的Comparator进行排序

参考代码:

package com.itheima_03;

import java.util.ArrayList;

/*
Stream<T> sorted():返回由此流的元素组成的流,根据自然排序

Stream<T> sorted(Comparator comparator):返回由该流的元素组成的流,根据提供的Comparator进行排序
 */
public class StreamDemo04 {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("lingqingxia");
        list.add("zhangmanyu");
        list.add("wangzuxian");
        list.add("liuyan");
        list.add("zhangmin");
        list.add("zhangwuji");

        //需求1:按照字母顺序把数据在控制台输出
//        list.stream().sorted().forEach(System.out::println);

        //需求2:按照字符串长度把数据在控制台输出
        //当长度相同时,没有在做处理
//        list.stream().sorted((s1, s2) -> s1.length() - s2.length()).forEach(System.out::println);

        list.stream().sorted((s1, s2) -> {
            int num = s1.length() - s2.length();    //先比较长度
            int num2 = num == 0 ? s1.compareTo(s2) : num;   //如果长度相等就比较自然字母顺序
            return num2;
        }).forEach(System.out::println);
    }
}

就是将流中的数据依次进行某个操作,这个操作由Function接口的具体实现来进行

Stream map(Function mapper):返回由给定函数应用于此流的元素的结果组成的流

​ Function接口中的方法 R apply(T t)

IntStream map ToInt(ToIntFunction mapper):返回一个IntStream其中包含将给定函数应用于此流的元素的结果

​ InStream:表示原始的int流

​ ToIntFunction接口中的方法 int applyAsInt(T value)

参考代码:

package com.itheima_03;

import java.util.ArrayList;

/*
<R> Stream<R> map(Function mapper):返回由给定函数应用于此流的元素的结果组成的流

​		Function接口中的方法		R apply(T t)

IntStream map ToInt(ToIntFunction mapper):返回一个IntStream其中包含将给定函数应用于此流的元素的结果

​		InStream:表示原始的int流

​		ToIntFunction接口中的方法		int applyAsInt(T value)
 */
public class StreamDemo05 {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("10");
        list.add("20");
        list.add("30");
        list.add("40");
        list.add("50");

        //需求:将集合中的字符串数据转换为整数之后在控制台输出
//        list.stream().map(s -> Integer.parseInt(s)).forEach(System.out::println);
//        list.stream().map(Integer::parseInt).forEach(System.out::println);    //使用方法引用


        //通过mapToInt方法获取IntStream流,就可以使用其中特有的方法
        list.stream().mapToInt(Integer::parseInt).forEach(System.out::println);

        //int sum() 返回此流中元素的总和。 IntStream接口中的一个方法
        int result = list.stream().mapToInt(Integer::parseInt).sum();
        System.out.println(result);
    }
}

四、Stream流常见的终结操作

Stream流的常见终结操作方法:

​ void forEach(Consumer action):对此流的每一个元素执行操作

​ Consumer接口中的方法 void accept(T t):对给定的参数执行此操作

​ long count():返回此流中的元素个数

参考代码:

package com.itheima_04;

import java.util.ArrayList;

/*
    Stream流的常见终结操作方法:

​		void forEach(Consumer action):对此流的每一个元素执行操作
​				Consumer接口中的方法		void accept(T t):对给定的参数执行此操作
​		long count():返回此流中的元素个数
 */
public class StreamDemo {
    public static void main(String[] args){
        //创建一个集合,存储多个字符串元素
        ArrayList<String> list = new ArrayList<String>();

        list.add("林青霞");
        list.add("张曼玉");
        list.add("王祖贤");
        list.add("柳岩");
        list.add("张敏");
        list.add("张无忌");

        //需求1:把集合中的元素在控制台输出
//        list.stream().forEach(System.out::println);

        //需求2:统计集合中有几个以张开头的元素,并把统计结果在控制台输出
        long count = list.stream().filter(s -> s.startsWith("张")).count();
        System.out.println(count);
    }
}

五、Stream流的练习

现在有两个ArrayList集合,分别存储6名男演员名称和6名女演员名称,要求完成如下操作:

​ 男演员只要名字为三个字的前三人

​ 女演员只要姓林的,并且不要第一个

​ 把过滤后的男演员姓名和女演员姓名合并到一起

​ 把上一步操作后的元素作为构造方法的参数创建演员对象,遍历数据

​ 演员类Actor已经提供,里面有一个成员变量,一个带参构造方法,以及成员变量对应的get/set方法

参考代码:

package com.itheima_05;

import java.util.ArrayList;
import java.util.stream.Stream;

public class StreamTest {
    public static void main(String[] args){
        //创建集合
        ArrayList<String> manList = new ArrayList<String>();
        manList.add("周润发");
        manList.add("成龙");
        manList.add("刘德华");
        manList.add("吴京");
        manList.add("周星驰");
        manList.add("李连杰");

        ArrayList<String> womanList = new ArrayList<String>();
        womanList.add("林心如");
        womanList.add("张曼玉");
        womanList.add("林青霞");
        womanList.add("柳岩");
        womanList.add("林志玲");
        womanList.add("王祖贤");

       /* //男演员只要名字为三个字的前三人
        Stream<String> manStream = manList.stream().filter(s -> s.length() == 3).limit(3);

        //女演员只要姓林的,并且不要第一个
        Stream<String> womanStream = womanList.stream().filter(s -> s.startsWith("林")).skip(1);

        //把过滤后的男演员姓名和女演员姓名合并到一起
        Stream<String> stream = Stream.concat(manStream, womanStream);

        //把上一步操作后的元素作为构造方法的参数创建演员对象,遍历数据
//        stream.map(Actor::new).forEach(System.out::println);  //没有重写toString方法

        //forEach中的泛型变为演员类,并通过对象来调用对应的getName方法
        stream.map(Actor::new).forEach(p -> System.out.println(p.getName()));*/

        //合并写法:
        Stream.concat(manList.stream().filter(s -> s.length() == 3).limit(3),
                womanList.stream().filter(s -> s.startsWith("林")).skip(1)).map(Actor::new)
        .forEach(p -> System.out.println(p.getName()));
    }
}

六、Stream流的收集操作

对数据使用Stream流的方式操作完毕后,我想把流中的数据收集到集合中,怎么办?

Stream流的收集方法:

​ R collect(Collectior collectior)

​ 但是这个收集方法的参数是一个Collector接口

工具类Collectors提供了具体的收集方式

​ public static Collector toList():把元素收集到List集合中

​ public static Collector toSet():把元素收集到Set集合中

​ public static Collector toMap(Function keyMapper, Function valueMapper):把元素收集到Map集合中

参考代码:

package com.itheima_06;

import java.util.*;
import java.util.stream.Collectors;
import java.util.stream.Stream;

/*
Stream流的收集方法:
​		R collect(Collectior collectior)
​		但是这个收集方法的参数是一个Collector接口

工具类Collectors提供了具体的收集方式
​		public static <T> Collector toList():把元素收集到List集合中
​		public static <T> Collector toSet():把元素收集到Set集合中
​		public static Collector toMap(Function keyMapper, Function valueMapper):把元素收集到Map集合中
 */
public class CollectDemo {
    public static void main(String[] args){
        //创建List集合对象
        ArrayList<String> list = new ArrayList<String>();
        list.add("林青霞");
        list.add("张曼玉");
        list.add("王祖贤");
        list.add("柳岩");

        /*//需求1:得到名字为3个字的流
        Stream<String> listStream = list.stream().filter(s -> s.length() == 3);

        //需求2:把使用Stream流操作完毕的数据收集到List集合中并遍历
        List<String> names = listStream.collect(Collectors.toList());
        for(String name : names){
            System.out.println(name);
        }*/

        //创建Set集合对象
        Set<Integer> set = new HashSet<Integer>();
        set.add(10);
        set.add(20);
        set.add(30);
        set.add(33);
        set.add(35);

        /*//需求3:得到年龄大于25的流
        Stream<Integer> setStream = set.stream().filter(age -> age > 25);

        //需求4:把使用Stream流操作完毕的数据收集到Set集合中并遍历
        Set<Integer> ages = setStream.collect(Collectors.toSet());
        for(int age : ages){
            System.out.println(age);
        }*/


        //定义一个字符串数组,每一个字符串数据由姓名数据和年龄数据组合而成
        String[] strArray = {"林青霞,30", "张曼玉,35", "王祖贤,33", "柳岩,25"};

        //需求5:得到字符串中年龄数据大于28的流
        Stream<String> arrayStream = Stream.of(strArray).filter(s -> Integer.parseInt(s.split(",")[1]) > 28);

        //需求6:把使用Stream流操作完毕后的数据收集到Map集合中并遍历,字符串中的姓名作为键,年龄作为值
        Map<String, Integer> map = arrayStream.collect(Collectors.toMap(s -> s.split(",")[0],
                s -> Integer.parseInt(s.split(",")[1])));

        for(String key : map.keySet()){
            System.out.println(key + "," + map.get(key));
        }

    }
}

标签:Java,Stream,stream,list,29,System,add,println
From: https://www.cnblogs.com/fragmentary/p/17005500.html

相关文章

  • 为什么阿里的Java开发规范中禁止使用Executors创建线程池?
    一.问题概述最近壹哥有个学生出去面试,面试官的一个问题是:在开发中你使用什么方式创建线程池?这个学生答曰:使用jdk中自带的工厂类Executors创建线程池!该学生回答完问题后,感......
  • 高薪程序员&Java面试题精讲系列汇总
    因为没有分类归纳博客的功能,所以特写本帖汇总《高薪程序员面试题精讲系列教程》,方便大家查阅!希望各位小伙伴,可以从我的拙作中能对Java的高频面试题有所掌握,也希望各位可以多......
  • Linux系统上Java单体项目崩溃自启动脚本(通用型)
    本文转自https://blog.csdn.net/qq_38374397/article/details/127566529实现对进程的监控这里只需要三步:设置服务启动脚本设置监控shell脚本设置linux周期定时执行指令......
  • Java代码打包
    Java代码打包一:IDEA工具1、右侧的maven直接clean后package2、终端命令mvncleanpackage3、项目结构,创建工件,选择清单属性,构建工件4、指定主类打包、含多个类的jar打......
  • Java基础之常用类(String类)
    String类定义String类代表字符串。Java程序中的所有字符串字面值(如"abc")都作为此类的实例实现。我们可以将字符串看作是String,但是严格意义上来说,String还是......
  • javaScript 列表常用语法基础大全
    javascript数组常用方法1.push()=>语法,数组.push(数据)=>作用:向数组的末尾追加数据=>返回值:添加数据以后,返回新的数组2.pop()=>语法,数组.pop(数据)=>作用:删除......
  • Java:泛型方法、泛型类、泛型接口、类型通配符
    (目录)要求:JDK>=1.5泛型方法packagecom.example.demo;importjava.io.IOException;publicclassDemo{//泛型方法publicstatic<T>voidprintT(T......
  • AcWing291.蒙德里安的梦想题解
    题解:蒙德里安的梦想注:本题解内容简陋,多有不周,敬请谅解。如果有问题请在评论区留言。谢谢。由于作者能力有限,这篇题解不会给出太严谨的证明,只是旨在帮助大家更好地理解此......
  • Java编程思想17
    第二十一章:并发基本的线程机制  并发编程使我们可以将程序划分为多个分离的、独立运行的任务。通过使用多线程机制,这些独立任务(也被称为子任务)中的每一个都将由执行线程......
  • Java编程思想18
    从任务种产生返回值:Runnable是执行工作的独立任务,但是它不返回任何值。如果你希望在任务完成时能够返回一个值,那么可以实现Callable接口而不是Runnable接口。在JavaSE5中......