29_Java中的Stream流
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
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
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
Stream
参考代码:
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
参考代码:
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
Stream
参考代码:
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接口的具体实现来进行
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
public static
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)); } } }
本文作者:如此而已~~~
本文链接:https://www.cnblogs.com/fragmentary/p/17005500.html
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步