1.Stream
流
1.体验Stream
流
需求:按照下面的要求完成集合的创建和遍历
- 创建一个集合,存储多个字符串元素
- 把集合中所有以“张”开头的元素存储到一个新的集合
- 把“张”开头的集合中的长度为3的元素存储到一个新的集合
- 遍历上一步得到的集合
代码:
package Study05;
import java.util.ArrayList;
/*
需求:按照下面的要求完成集合的创建和遍历
- 创建一个集合,存储多个字符串元素
- 把集合中所有以“张”开头的元素存储到一个新的集合
- 把“张”开头的集合中的长度为3的元素存储到一个新的集合
- 遍历上一步得到的集合
*/
public class StreamDemo {
public static void main(String[] args) {
//- 创建一个集合,存储多个字符串元素
ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("张小胖");
arrayList.add("张春华");
arrayList.add("张胖虎");
arrayList.add("李小乐");
arrayList.add("张华");
//- 把集合中所有以“张”开头的元素存储到一个新的集合
ArrayList<String> zhangList = new ArrayList<String>();
for (String s:arrayList){
if (s.startsWith("张")){
zhangList.add(s);
}
}
//- 把“张”开头的集合中的长度为3的元素存储到一个新的集合
ArrayList<String> zhang3List = new ArrayList<String>();
for (String s:zhangList){
if (s.length()==3){
zhang3List.add(s);
}
}
//- 遍历上一步得到的集合
for (String s:zhang3List){
System.out.println(s);
}
System.out.println("====================================");
//采用Stream流来改写
// arrayList.stream().filter(s->s.startsWith("张")).filter(s->s.length()==3).forEach(s-> System.out.println(s));
arrayList.stream().filter(s->s.startsWith("张")).filter(s->s.length()==3).forEach(System.out::println);
}
}
使用Stream流的方式完成过滤操作
-
arrayList.stream().filter(s->s.startsWith("张")).filter(s->s.length()==3).forEach(System.out::println);
-
直接阅读代码的字面意思即可完美展示无关逻辑方式的语义:生成流,过滤张,过滤长度为3,逐一打印
-
Stream
流把真正的函数式编程风格引入到Java
中
2.Stream
流概述
Stream
流的使用
- 生成流
- 通过数据源(集合,数组等)生成流
list.stream()
- 中间操作
- 一个流后面可以跟随零个或多个中间操作,其目的主要是打开流,做出某种程度的数据过滤/映射,然后返回一个新的流,交给下一个操作使用
filter()
- 终结操作
- 一个流只能有一个终结操作,当这个操作执行后,流就被使用“光”了,无法再被操作。所以这必定是流的最后一个操作
forEach()
2.Stream
流的常见生成方式
Stream
流的常见生成方式
Collection
体系的集合可以使用默认方法stream()
生成流default Stream <E> stream()
Map
体系的集合间接生成流- 数组可以通过
Stream
接口的静态方法of(T…values)
生成流
代码示例:
package Study06;
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流;map.keySet()返回的是Set类型的集合
Stream<Integer> valuesStream = map.values().stream();//值的Stream流;map.values()返回的是Collection类型的集合
Stream<Map.Entry<String, Integer>> entryStream = map.entrySet().stream();//键值对的Stream流; map.entrySet()返回的是Set类型的集合
//- 数组可以通过`Stream`接口的静态方法`of(T…values)`生成流
String[]strArray ={"迪丽热巴","王冰冰"};
Stream<String> strArrayStream = Stream.of(strArray);
Stream<String> strArrayStream2 = Stream.of("迪丽热巴", "王冰冰");
Stream<Integer> integerStream = Stream.of(10, 20, 30);
}
}
3.Stream
流的常见中间操作方法
-
Stream filter(Predicate predicate):用于对流中的数据进行过滤。
-
Predicate接口中的方法:
- boolean test(T t):对给定的参数进行判断,返回一个布尔值
-
Stream<T>limit (long maxSize)
:返回此流中的元素组成的流,截取前指定参数个数的数据 -
Stream<T> skip(long n)
:跳过指定参数个数的数据,返回由该流的剩余元素组成的流 -
static<T> Stream<T> concat(Stream a,Stream b)
:合并a和b两个流为一个流。 -
Stream<T> distinct()
:返回由该流的不同元素(根据Object.equals(Object)
)组成的流 -
Stream<T> sorted()
:返回由此流的元素组成的流,根据自然顺序排序 -
Stream<T> sorted(Comparator comparator):
返回由该流的元素组成的流,根据提供的Comparator
进行排序 -
<R> Stream <R>map(Function mapper)
:返回由给定函数应用于此流的元素的结果组成的流Function
接口的方法:R apply(T t)
-
IntStream mapToInt(TOIntFunction mapper):
返回一个IntStream
其中包含给定函数应用于此流的元素的结果IntStream
:表示原始int流ToIntFunction
接口的方法:int applyAsInt(T value)
1.filter
- Stream filter(Predicate predicate):用于对流中的数据进行过滤。
- Predicate接口中的方法:
- boolean test(T t):对给定的参数进行判断,返回一个布尔值
代码:
package Study06;
import java.util.ArrayList;
/*
- Stream<T> filter(Predicate predicate):用于对流中的数据进行过滤。
- Predicate接口中的方法:
- boolean test(T t):对给定的参数进行判断,返回一个布尔值
*/
public class StreamDemo01 {
public static void main(String[] args) {
//- 创建一个集合,存储多个字符串元素
ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("张小胖");
arrayList.add("张春华");
arrayList.add("张胖虎");
arrayList.add("李小乐");
arrayList.add("张华");
//需求1:把集合中所有以“张”开头的元素在控制台输出
arrayList.stream().filter(s->s.startsWith("张")).forEach(System.out::println);
System.out.println("----------------");
//需求2:把“张”开头的集合中的长度为3的元素在控制台输出
arrayList.stream().filter(s->s.length()==3).forEach(System.out::println);
System.out.println("----------------");
//需求3:把“张”开头的集合中的长度为3的元素在控制台输出
arrayList.stream().filter(s->s.startsWith("张")).filter(s->s.length()==3).forEach(System.out::println);
System.out.println("----------------");
}
}
2.limit
&&skip
Stream<T>limit (long maxSize)
:返回此流中的元素组成的流,截取前指定参数个数的数据Stream<T> skip(long n)
:跳过指定参数个数的数据,返回由该流的剩余元素组成的流
代码示例:
package Study06;
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> arrayList = new ArrayList<String>();
arrayList.add("张小胖");
arrayList.add("张春华");
arrayList.add("张胖虎");
arrayList.add("李小乐");
arrayList.add("张华");
//需求1:取前三个数据在控制台输出
arrayList.stream().limit(3).forEach(System.out::println);
System.out.println("-----------");
//需求2:跳过3个元素,把剩下的元素在控制台输出
arrayList.stream().skip(3).forEach(System.out::println);
System.out.println("-----------");
//需求3:跳过2个元素,把剩下的元素中前2个在控制台输出
arrayList.stream().skip(2).limit(2).forEach(System.out::println);
System.out.println("-----------");
}
}
3.concat
&&distinct
static<T> Stream<T> concat(Stream a,Stream b)
:合并a和b两个流为一个流。Stream<T> distinct()
:返回由该流的不同元素(根据Object.equals(Object)
)组成的流
代码示例:
package Study06;
import java.util.ArrayList;
import java.util.stream.Stream;
/*
- `static<T> Stream<T> concat(Stream a,Stream b)`:合并a和b两个流为一个流。
- `Stream<T> distinct()`:返回由该流的不同元素(根据`Object.equals(Object)`)组成的流
*/
public class StreamDemo03 {
public static void main(String[] args) {
//- 创建一个集合,存储多个字符串元素
ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("张小胖");
arrayList.add("张春华");
arrayList.add("张胖虎");
arrayList.add("李小乐");
arrayList.add("张华");
//需求1:取前4个数据组成一个流
Stream<String> limit = arrayList.stream().limit(4);
//需求2:跳过2个数据组成一个流
Stream<String> skip = arrayList.stream().skip(2);
//需求3:合并需求1和需求2得到的流,并把结果在控制台输出
// Stream.concat(limit,skip).forEach(System.out::println);
System.out.println("-----------");
//需求4:合并需求1和需求2得到的流,并把结果在控制台输出,要求字符串元素不能重复
Stream.concat(limit,skip).distinct().forEach(System.out::println);
}
}
4.sorted
Stream<T> sorted()
:返回由此流的元素组成的流,根据自然顺序排序Stream<T> sorted(Comparator comparator):
返回由该流的元素组成的流,根据提供的Comparator
进行排序
代码示例:
package Study06;
import java.util.ArrayList;
import java.util.Comparator;
/*
- `Stream<T> sorted()`:返回由此流的元素组成的流,根据自然顺序排序
- `Stream<T> sorted(Comparator comparator):`返回由该流的元素组成的流,根据提供的`Comparator`进行排序
*/
public class StreamDemo04 {
public static void main(String[] args) {
//- 创建一个集合,存储多个字符串元素
ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("zhangxiaopang");
arrayList.add("zhangchunhua");
arrayList.add("zhangpanghu");
arrayList.add("lixiaole");
arrayList.add("zhanghua");
//需求1:按照字符顺序把数据在控制台输出
// arrayList.stream().sorted().forEach(System.out::println);
//需求2:按照字符串长度把数据在控制台输出
// arrayList.stream().sorted((s1, s2) -> s1.length() - s2.length()).forEach(System.out::println);//当长度相同时,先输入的在前
arrayList.stream().sorted((s1,s2)->{
int num = s1.length()-s2.length();
int num1 = num==0?s1.compareTo(s2):num;
return num1;
}).forEach(System.out::println);//当长度相同时,按字符顺序进行排序
}
}
5.map
&&mapToInt
<R> Stream <R>map(Function mapper)
:返回由给定函数应用于此流的元素的结果组成的流Function
接口的方法:R apply(T t)
IntStream mapToInt(TOIntFunction mapper):
返回一个IntStream
其中包含给定函数应用于此流的元素的结果IntStream
:表示原始int流ToIntFunction
接口的方法:int applyAsInt(T value)
代码示例:
package Study06;
import java.util.ArrayList;
/*
- `<R> Stream <R>map(Function mapper)`:返回由给定函数应用于此流的元素的结果组成的流
- `Function`接口的方法:`R apply(T t)`
- `IntStream mapToInt(TOIntFunction mapper):`返回一个`IntStream`其中包含给定函数应用于此流的元素的结果
- `IntStream`:表示原始int流
- `ToIntFunction`接口的方法:`int applyAsInt(T value)`
*/
public class StreamDemo05 {
public static void main(String[] args) {
//- 创建一个集合,存储多个字符串元素
ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("10");
arrayList.add("20");
arrayList.add("30");
arrayList.add("40");
arrayList.add("50");
//需求:将集合中的字符数据转换为整数类型的数据并在控制台输出
// arrayList.stream().map(s->Integer.parseInt(s)).forEach(System.out::println);
// arrayList.stream().map(Integer::parseInt).forEach(System.out::println);
// arrayList.stream().mapToInt(Integer::parseInt).forEach(System.out::println);
//int sum():返回此流中元素的总和
int sum = arrayList.stream().mapToInt(Integer::parseInt).sum();
System.out.println(sum);
}
}
4.Stream
流的常见终结操作方法
Stream流的常见终结操作
- void forEach(Consumer action):对此流的每个元素执行操作
- Consumer接口中的方法
- void accept(T t):对给定的参数执行操作
- Consumer接口中的方法
- long count():返回此流中的元素数
代码示例:
package Study06;
import java.util.ArrayList;
/*
Stream流的常见终结操作
- void forEach(Consumer action):对此流的每个元素执行操作
- Consumer接口中的方法
- void accept(T t):对给定的参数执行操作
- long count():返回此流中的元素数
*/
public class StreamDemo06 {
public static void main(String[] args) {
//- 创建一个集合,存储多个字符串元素
ArrayList<String> arrayList = new ArrayList<String>();
arrayList.add("张小胖");
arrayList.add("张春华");
arrayList.add("张胖虎");
arrayList.add("李小乐");
arrayList.add("张华");
//需求1:把集合中的元素在控制台输出
arrayList.stream().forEach(System.out::println);
System.out.println("-----------");
//需求2:统计集合中以张开头的元素,统计结果在控制台输出
long count = arrayList.stream().filter(s -> s.startsWith("张")).count();
System.out.println(count);
}
}
5.Stream
流的练习
需求:
现有两个ArrayList
集合,分别存储6名男生名称和6名女生名称,要求完成如下操作:
- 男生只要名字为三个字的前三人
- 女生只要姓林的,并且不要第一个
- 把过滤后的男生和女生姓名合并在一起
- 把上一步操作后的元素作为构造方法的参数穿件演员对象,遍历数据
- 演员类
Actor
:有一个成员变量,一个带参构造方法,以及成员变量对应的get/set方法
- 演员类
代码:
Actor
类:
package Study06;
public class Actor {
private String name;
public Actor() {
}
public Actor(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
StreamTest
测试类:
package Study06;
import java.util.ArrayList;
import java.util.stream.Stream;
/*需求:
现有两个`ArrayList`集合,分别存储6名男生名称和6名女生名称,要求完成如下操作:
- 男生只要名字为三个字的前三人
- 女生只要姓林的,并且不要第一个
- 把过滤后的男生和女生姓名合并在一起
- 把上一步操作后的元素作为构造方法的参数穿件演员对象,遍历数据
- 演员类`Actor`:有一个成员变量,一个带参构造方法,以及成员变量对应的get/set方法
*/
public class StreamTest {
public static void main(String[] args) {
ArrayList<String>boy=new ArrayList<String>();
boy.add("懒洋洋");
boy.add("灰太狼");
boy.add("沸羊羊");
boy.add("村长");
boy.add("喜洋洋");
ArrayList<String>girl=new ArrayList<String>();
girl.add("林琳");
girl.add("林颖");
girl.add("张素华");
girl.add("林志玲");
/* //- 男生只要名字为三个字的前三人
Stream<String> boy3 = boy.stream().filter(s->s.length()==3).limit(3);
//- 女生只要姓林的,并且不要第一个
Stream<String> girl1 = girl.stream().filter(s->s.startsWith("林")).skip(1);
//- 把过滤后的男生和女生姓名合并在一起
Stream<String> BoyAndGirl = Stream.concat(boy3, girl1);
//- 把上一步操作后的元素作为构造方法的参数创建演员对象,遍历数据
// BoyAndGirl.map(Actor::new).forEach(System.out::println);
BoyAndGirl.map(Actor::new).forEach(p-> System.out.println(p.getName()));*/
//合并上述操作
Stream.concat(boy.stream().filter(s->s.length()==3).limit(3),
girl.stream().filter(s->s.startsWith("林")).skip(1)).
map(Actor::new).forEach(p-> System.out.println(p.getName()));
}
}
6.Stream
流的收集操作
对数据使用Stream
流的方式操作完毕后,如何将流中的数据收集到集合中?
Stream
流的搜集方法
R collect(Collector collector)
- 这个收集方法的参数是一个
Collector
接口
工具类Collectors
提供了具体的搜集方式
public static <T>Collection toList():
把元素收集到List
集合中public static <T>Collector toSet()
:把元素收集到Set
集合中public static Collector toMap(Function keyMapper,Function valueMapper)
:把元素收集到Map
集合中
示例代码:
package Study06;
import java.util.*;
import java.util.stream.Collectors;
import java.util.stream.Stream;
/*`Stream`流的搜集方法
- `R collect(Collector collector)`
- 这个收集方法的参数是一个`Collector`接口
工具类`Collectors`提供了具体的搜集方式
- `public static <T>Collection 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) {
ArrayList<String> list = new ArrayList<String>();
list.add("懒洋洋");
list.add("灰太狼");
list.add("沸羊羊");
list.add("村长");
list.add("喜洋洋");
//- `public static <T>Collection toList():`把元素收集到`List`集合中
/*//需求1:得到名字为3个字的流
Stream<String> listStream = list.stream().filter(s -> s.length() == 3);
//需求2:把使用Stream流操作完毕的数据收集到List集合中并遍历
List<String> names = listStream.collect(Collectors.toList());
for (String s:names){
System.out.println(s);
}*/
//- `public static <T>Collector toSet()`:把元素收集到`Set`集合中
//创建Set集合
Set<Integer> set = new HashSet<Integer>();
set.add(10);
set.add(20);
set.add(30);
set.add(40);
/*//需求3:得到年龄大于20的流
Stream<Integer> ages = set.stream().filter(s -> s > 20);
//需求4:把使用Stream流操作完毕的数据收集到Set集合中并遍历
Set<Integer> age = ages.collect(Collectors.toSet());
for (Integer i : age) {
System.out.println(i);
}*/
//定义一个字符串数组,每一个字符串数据由姓名数据和年龄数据组合而成
String[] str = {"喜洋洋,10","懒洋洋,20","美洋洋,30","沸羊羊,40","村长,50"};
//需求5:得到字符串中年龄大于20的流
Stream<String> stringStream = Stream.of(str).filter(s -> Integer.parseInt(s.split(",")[1]) > 20);
//需求6:把使用Stream流操作完毕的数据收集到Map集合并遍历,字符串中的姓名作键,年龄作值
Map<String, Integer> map = stringStream.collect(Collectors.toMap(s -> s.split(",")[0], s -> Integer.parseInt(s.split(",")[1])));
Set<String> key = map.keySet();
for (String k:key){
System.out.println("姓名:"+k+";年龄:"+map.get(k));
}
}
}