java中重要的函数接口
接口 | 参数 | 返回类型 | 示例 |
---|---|---|---|
Predicate< T > | T | boolean | 这张唱片已经发行了吗 |
Consumer< T > | T | void | 输出一个值 |
Function< T,R > | T | R | 获得Artist对象的名字 |
Supplier< T > | None | T | 工厂方法 |
UnaryOperator< T > | T | T | 逻辑非 |
BinaryOperator< T > | (T,T) | T | 求两个数的乘积 |
stream 中的 filter 方法和 count 方法
filter 方法并未做什么实际性的工作,只刻画出了stream,但没有产生新的集合,这类方法叫惰性求值;
count 方法最终会从 stream 产生值,这类方法叫及早求值方法。
Stream.of()和*.stream()的区别
Stream.of(T t...)中t是流的多个元素,.stream()是把集合的每个元素变成流
Stream<List<Track>> stream = Stream.of(tracks);
Stream<Track> stream2 = tracks.stream();
-
map
元素处理,一一映射List<Integer> collect1 = allInt.stream().filter(ele -> ele > 5)
.map(ele -> ele * 10).collect(Collectors.toList());
System.out.println(collect1);
- filter
过滤元素 - flatMap
可以把多个流合并成一个流
返回值是stream
List<Integer> collect = Stream.of(list1, list2).flatMap(ele -> ele.stream())
.collect(Collectors.toList());
System.out.println(collect);
- min() & max()
比较
List<Track> tracks = Arrays.asList(new Track("imagine", 92),new Track
("super star", 28), new Track("halo", 64));
Track track1 = tracks.stream().min(Comparator.comparing(track ->
track.getName().length())).get();
- reduce
循环计算
reduce(初始值, BinaryOperator<T>(传递的循环计算结果, 遍历的流中的值))
Integer reduce = Stream.of(1, 2, 3).reduce(10, (acc, element) -> acc - element);
拆装箱
int => Integer : intFunction
Integer => int : toIntFunction
高阶函数使用基本类型:mapToInt()——方法 + to + 基本类型
mapToInt 返回的不是一个一般的 stream,这个 stream 的 map() 等方法都用的不是原来的接口。
该特殊 stream 有个 summaryStatistics() 方法得到的对象中有诸多计算属性,方便计算。
IntSummaryStatistics iss = album.getTracks().stream()
.mapToInt(trac -> trac.getLength())
.map(length -> length + 1)
.summaryStatistics();
System.out.println(iss);
// 输出
IntSummaryStatistics{count=1, sum=11, min=11, average=11.000000, max=11}
重载
会优先使用最具体的类型的函数
错误: Ambiguous method call .Both
重载函数模糊调用,参数列表都匹配。
- 只有一个可能的目标类型:由相应函数接口里的参数类型推导得出;
- 多个可能的目标类型:由最具体的参数类型推导得出;
- 多个可能的目标且具体类型不明:需人为指定类型,否则编译报错。
默认方法
接口(无论函数还是非函数)告诉它的所有子类:如果你没有实现这个方法,就使用我的吧。
子类重写方法覆盖默认方法。
如果一个类实现了两个含有相同签名函数的接口,编译报错。解决方法:实现类实现该函数。
- 类胜于接口
- 子类胜于父类
接口:允许多重继承,没有成员变量;
抽象类:不能多重继承,可继承成员变量。
Optional
String s = null;
Optional<String> optional = Optional.ofNullable(s);
s = "aa";
在上面这段代码中,最后一行赋值并没有任何作用:Optional对象直接被赋予了String的值,而非String对象
Optional对空值更简单的处理:
Optional<String> op = Optional.ofNullable(str);
String result = op.orElseGet(() -> "");
相当于:
String result = str==null ? "" : str;
orElse 和 orElseGet 的区别
optional.orElse("unknow");
optional.orElseGet(() -> "unknow");
流的顺序
原集合有序,则出去的流依然有序;原集合无序,则出去的流也无序。
一些操作在有序的流上开销更大,可调用unordered方法消除顺序;大多数操作(filter map reduce)在有序流上效率更高。
Collect
stream类库在collect的时候自动挑选合适的集合类型。
如果需要指定类型,则可以:
stream.collect(Collectors.toCollection(TreeSet::new));
转换成值
// 找出成员最多的乐队
Function<Artist, Long> getCount = artist -> artist.getMembers.count();
Optional<Artist> maxArtist = artists.stream().
collect(Collectors.maxBy(
Comparator.comparing(getCount)));
// 一组乐队的成员平均数
Double averageMember = artists.stream().collect(
Collectors.averagingInt(
artist -> artist.getMembers().size()));
partitioningBy收集器
// 区分乐队和单个歌手
Map<Boolean,List<Artist>> map =
stream.collect(Collectors.partitioningBy
(artist -> artist.isSolo()));
// 另一种写法:使用方法
Map<Boolean, List<Artist>> map = stream.collect(
Collectors.partitioningBy(Artist:: isSolo));
拼接字符串
把流中的元素按照规律拼接成字符串。
joining(元素中分隔字符,开始字符,结束字符)
String collect = f4.getMembers().stream()
.map(Artist:: getName)
.collect(Collectors.joining(",","[","]"));
组合收集器
把流中的元素按照规则分组,并与想收集的值关联。
groupingBy(分组的方式Function, 收集的数据Collector)
// 计算每个艺术家的专辑数
Map<Artist, Long> map = albums.stream().collect(
Collectors.groupingBy(Album:: getMainArtist, Collectors.counting()));
// 每个艺术家的专辑名
Map<Artist, List<String>> map = albums.stream().collect(
Collectors.groupingBy(Album:: getMainArtist,
Collectors.mapping(Album:: getName, Collectors.toList())));