我了解了java lambda表达式和函数式接口之后,就经常找机会在实战中使用它们。但是仅仅只了解这两者,在很多场景下都无法方便地使用。很多支持函数式编程的语言,都提供一些高级操作(高阶函数),如map和reduce。
java.util.stream package正是出于此目的,引入了一系列高级API,加强函数式编程在实战的实用性,也使得原本复杂的控制代码变得简洁明了。本文是一个刚刚了解函数式编程的新手对stream api的初步总结。
1.元素序列上的操作
如果不了解stream api,那么在进行序列操作时,只能采用循环,如下。
//Collection<E> c;
for(E e : c){
//option
}
学习了java stream api后,你可以这么做:
c.stream().forEach(e->{//option
})
但是,这不还是迭代吗?有区别吗?
区别还是有的,
1.可以使用lambda表达式简化代码
2.stream底层实现支持并行处理,大大提高了处理的性能。这也是stream的强大之处
当然,很多时候循环体内的操作很复杂,简单的lambda不足以表达。所以stream package还支持了pipeline operation。我理解为流水线操作。例子如下:
//求一群人中未成年人的最大年龄
//List<Person> lp;
lp.stream().mapToInt(()->lp.getAge()).filter(i->i<18).max();
上面mapToInt函数将源stream转化,生成一个新的IntStream,然后筛选出其中的未成人年龄,最后取其中最大值。
这个过程将复杂操作步步拆分,再分别表达,让代码变得非常简洁。原来可能需要多层循环,多处判断的地方,现在可以很直观使用方法进行表意。
2.常用操作
Stream类中包含了对stream的操作。有特定操作,通用操作,以及生成stream的方法,分别简单介绍如下:
//预先定义数据
class Student{
public String name;
public int class;
public String major;
}
List<Student> ls = new ArrayList<Student>();
特定操作
元素匹配(allMatch,anyMatch,noneMatch)
//判断是否所有学生主修计算机
ls.stream().allMatch(s->s.major.equals("computer"));
//判断其中是否有7班的学生
ls.stream().anyMatch(s->s.class==7);
//判断其中是否有叫“Bob”的学生
ls.stream().noneMatch(s->s.name.equals("Bob"));
Optional<Student> f1 = li.stream().findAny();
Optional<Student> f2 = li.stream().findFirst();
筛选元素(filter)
//获取某个计算机系的学生
Optional<Student> s =
li.stream().filter(s->s.major.equals("computer")).findAny()
还有许多操作就不一一举例了,它们的方法名完全是自解释的,列举如下:
求最值(max,min),统计元素数量(count),删除重复元素(distinct),截断(limit,skip),排序(sorted)
通用操作
所谓通用操作就是,当特定api无法满足需求时,可以进行一定程度的定制化(策略模式)。但它本身也是存在模板的。
遍历(forEach,forEachOrdered,peek)
//打印所有学生的名字
ls.stream().forEach(s->System.out.println(s.name));
映射元素(map,flatMap),由stream推导最终结果(reduce,collect)
举例如下:
//统计7班人数个数
//方法一:使用特定api
ls.stream().filter(s->s.class==7).count()
//方法二:使用通用api
ls.stream().map(s->s.class==7?1:0).reduce(0,(sum,c)->sum+=c);
注:map,flatmap,reduce,collect均存在多个重在版本或者特化版本,需要依据实际情况选用。
生成stream
连接两个stream(concat)
获得空stream(empty)
指定元素生成stream(of)
根据指定的元素提供者生成stream(generate)
指定种子和函数迭代地生成stream(iterate)
3.特殊stream
DoubleStream,IntStream,LongStream
stream package对于double,int,long这三种基本数据类型提供了特殊的stream类,包含了stream类中方法的特化版本以及特殊的工厂方法。