前言
不知道从什么时候开始. Java8开始刮起了一阵妖风. 那就是lamda表达式. 也叫函数式编程, 大概是和隔壁的Scala进行学习吧. 但是Java8也是甲骨文最后一个免费的版本. 大概率, 很多的中小型公司都会选择停留在这个版本. 虽然这个版本没有ZGC垃圾回收, 但是面对10G-100G左右内存规模的程序, 也应该够了.
初识lamda表达式
lamda表达式最大的特性就是简化了for循环的编写方式. 多说无益, 我们举个例子进行编写吧.
List<Integer> helloArrayList = new ArrayList<>();
for(String str : helloArratList){
str = str+"a";
}
List<Integer> helloArrayList = new ArrayList<>();
// 注意lamda表达式中map处理完之后要使用collect收集. 并且赋值回去.
helloArrayList = helloArrayList.stream().map(str -> str+"a").collect(Collections.toList());
这样看起来是不是很简单? 好像也不是… 但是当多个算子聚合操作的时候, 你会感觉出来的. 特别其还有一些比较喜欢的Map转换器 .
map() 支持表达式的三种写法
差点忘了这一部分. 我们初学者写的时候非常苦恼. 其中的表达式有三种变种. 我们需要熟悉他们:
map(a -> {
a = a+1;
int. b. = 2;
a = a+b;
return a;
})
- map(String::charAt)
这一部分经常是动态的调用某个静态方法. 比如 filter(Objects::notNull) . 其实你不熟悉这种, 也无所谓. 能看懂即可.
三种写法是不是很像,孔乙己的回字的几种写法, 哈哈. 有种八股文的味道了.
Java8 lamda表达式算子介绍
在这里给出一些, 我在工作中使用比较多的算子.
-
map() 大名鼎鼎的map() 算子. map 通常的写法为xxx.stream().map(object -> objectxxxxx); 值得注意的是, 我初始的时候并不知道object 这个变量写什么好. 其实, 写多了之后, 会发现, 其可以任意指定. 其次, 当做业务开发时, 这个值最好和业务相关的. 比如bookArrayList.stream().map(book -> { return abc;}) -
collect() 这个算子中方法的变种比较多. 比如:
- collect(Collectors.toList()). 目标结果为
List<>abcList . - collect(Collectors.toMap(a.key ,a->a,(a1,a2)->a1)) . 目标结果为
Map<keyType, valueType> map . 这种需要注意map出现key重名情况. 获取key有时也有传递方法体的. 例如: collect(Collectors.toMap(Object::getHashCode ,a->a,(a1,a2)->a1)) collect(Collectors.toMap(a.key ,a->a,(a1,a2)->a1)) . - collect(Collectors.groupByMap(Object::getHashCode)). 目标结果为
Map<keyType, List<valueType>> . 注意, 这种是不需要考虑key重名的情况的. - collect(Collectors.toSet()). 目标结果为
Set<> abcSet .
更多值得注意的是. 之前和朋友提到过一个问题. 我们Collectors.toMap() 和 Collectors.groupByMap我们转换为map()的时候. 我们会出现key为null 或者为""(空字符串) 的情况么? 答: 答案很简单. 是会出现的. 那么转换为Set是否也会出现? 答案仍然是肯定的.
-
filter() 过滤器. 通常做一些筛选. 如果我要筛选不为null的情况. 那么表达式可以写为: filter(a -> null != a) /filter(a -> {return null!=a;}) / filter(Objects:notNull) . 注意这部都是只有bool为true时才能通过. -
foreach() 注意如果对于一个list来说. 你不想获取返回值. 那么可以直接forEach. 同样的对于Map类型来说, 也只有forEach方法.
Map<String, String> helloMap = new HashMap();
helloMap.forEach( (key,value) ->{
System.out.println(hellpMap);
});
其余不常用的算子
比较难用的点
个人在使用过程中, 难用的点有如下几种:
-
难以调试&难以阅读 如果你把所有的算子都写在一行的话. 那么读和调试都异常困难. 估计和你合作开发的人肯定会diss你, 哈哈. 个人推荐使用IDEA的Google的代码结构化换行工具. 这个前面的IDEA工具内有介绍过. 这边稍后会给个链接. -
有时不能赋值给某个非final变量 这个说的有点抽象. 等有合适的例子再补充在这里吧. -
无法使用下标获取. 当然如果想要下标, 还是需要老老实实写成for(int i-0; i<arrayList.length;i++){ arrayList.get(i);}
Reference
[1]. (菜鸟教程)Java 8 新特性 [2]. (菜鸟教程)Java 8 Stream
|