本文github地址
你可能没意识到Java对函数式编程的重视程度,看看Java 8加入函数式编程扩充多少功能就清楚了。Java 8之因此费这么大功夫引入函数式编程,缘由有二:html
parallel()
方法。这一节咱们学习stream,也就是Java函数式编程的主角。对于Java 7来讲stream彻底是个陌生东西,stream并非某种数据结构,它只是数据源的一种视图。这里的数据源能够是一个数组,Java容器或I/O channel等。正因如此要获得一个stream一般不会手动建立,而是调用对应的工具方法,好比:java
Collection.stream()
或者Collection.parallelStream()
方法Arrays.stream(T[] array)
方法常见的stream接口继承关系如图:git
图中4种stream接口继承自BaseStream
,其中IntStream, LongStream, DoubleStream
对应三种基本类型(int, long, double
,注意不是包装类型),Stream
对应全部剩余类型的stream视图。为不一样数据类型设置不一样stream接口,能够1.提升性能,2.增长特定接口函数。github
你可能会奇怪为何不把IntStream
等设计成Stream
的子接口?毕竟这接口中的方法名大部分是同样的。答案是这些方法的名字虽然相同,可是返回类型不一样,若是设计成父子接口关系,这些方法将不能共存,由于Java不容许只有返回类型不一样的方法重载。编程
虽然大部分状况下stream是容器调用Collection.stream()
方法获得的,但stream和collections有如下不一样:数组
对stream的操做分为为两类,中间操做(intermediate operations)和结束操做(terminal operations),两者特色是:数据结构
若是你熟悉Apache Spark RDD,对stream的这个特色应该不陌生。app
下表汇总了Stream
接口的部分常见方法:函数式编程
操做类型 | 接口方法 |
---|---|
中间操做 | concat() distinct() filter() flatMap() limit() map() peek() skip() sorted() parallel() sequential() unordered() |
结束操做 | allMatch() anyMatch() collect() count() findAny() findFirst() forEach() forEachOrdered() max() min() noneMatch() reduce() toArray() |
区分中间操做和结束操做最简单的方法,就是看方法的返回值,返回值为stream的大都是中间操做,不然是结束操做。函数
stream跟函数接口关系很是紧密,没有函数接口stream就没法工做。回顾一下:函数接口是指内部只有一个抽象方法的接口。一般函数接口出现的地方均可以使用Lambda表达式,因此没必要记忆函数接口的名字。
咱们对forEach()
方法并不陌生,在Collection
中咱们已经见过。方法签名为void forEach(Consumer<? super E> action)
,做用是对容器中的每一个元素执行action
指定的动做,也就是对元素进行遍历。
// 使用Stream.forEach()迭代 Stream<String> stream = Stream.of("I", "love", "you", "too"); stream.forEach(str -> System.out.println(str));
因为forEach()
是结束方法,上述代码会当即执行,输出全部字符串。
函数原型为Stream<T> filter(Predicate<? super T> predicate)
,做用是返回一个只包含知足predicate
条件元素的Stream
。
// 保留长度等于3的字符串 Stream<String> stream= Stream.of("I", "love", "you", "too"); stream.filter(str -> str.length()==3) .forEach(str -> System.out.println(str));
上述代码将输出为长度等于3的字符串you
和too
。注意,因为filter()
是个中间操做,若是只调用filter()
不会有实际计算,所以也不会输出任何信息。
函数原型为Stream<T> distinct()
,做用是返回一个去除重复元素以后的Stream
。
Stream<String> stream= Stream.of("I", "love", "you", "too", "too"); stream.distinct() .forEach(str -> System.out.println(str));
上述代码会输出去掉一个too
以后的其他字符串。
排序函数有两个,一个是用天然顺序排序,一个是使用自定义比较器排序,函数原型分别为Stream<T> sorted()
和Stream<T> sorted(Comparator<? super T> comparator)
。
Stream<String> stream= Stream.of("I", "love", "you", "too"); stream.sorted((str1, str2) -> str1.length()-str2.length()) .forEach(str -> System.out.println(str));
上述代码将输出按照长度升序排序后的字符串,结果彻底在预料之中。
函数原型为<R> Stream<R> map(Function<? super T,? extends R> mapper)
,做用是返回一个对当前全部元素执行执行mapper
以后的结果组成的Stream
。直观的说,就是对每一个元素按照某种操做进行转换,转换先后Stream
中元素的个数不会改变,但元素的类型取决于转换以后的类型。
Stream<String> stream = Stream.of("I", "love", "you", "too"); stream.map(str -> str.toUpperCase()) .forEach(str -> System.out.println(str));
上述代码将输出原字符串的大写形式。
函数原型为<R> Stream<R> flatMap(Function<? super T,? extends Stream<? extends R>> mapper)
,做用是对每一个元素执行mapper
指定的操做,并用全部mapper
返回的Stream
中的元素组成一个新的Stream
做为最终返回结果。提及来太拗口,通俗的讲flatMap()
的做用就至关于把原stream中的全部元素都"摊平"以后组成的Stream
,转换先后元素的个数和类型均可能会改变。
Stream<List<Integer>> stream = Stream.of(Arrays.asList(1,2), Arrays.asList(3, 4, 5)); stream.flatMap(list -> list.stream()) .forEach(i -> System.out.println(i));
上述代码中,原来的stream
中有两个元素,分别是两个List<Integer>
,执行flatMap()
以后,将每一个List
都“摊平”成了一个个的数字,因此会新产生一个由5个数字组成的Stream
。因此最终将输出1~5这5个数字。
截止到目前咱们感受良好,已介绍Stream
API理解起来并不费劲儿。若是你就此觉得函数式编程不过如此,恐怕是高兴地太早了。下一节对Stream
规约操做的介绍将刷新你如今的认识。
本文github地址,欢迎关注。