为引入Lambda表达式,Java8新增了java.util.funcion包,里面包含常用的函数接口,这是Lambda表达式的基础,Java集合框架也新增部分接口,以便与Lambda表达式对接。
首先回顾一下Java集合框架的接口继承结构:

上图中绿色标注的接口类,表示在Java8中加入了新的接口方法,当然由于继承关系,他们相应的子类也都会继承这些新方法。下表详细列举了这些方法:
| 接口名 | Java8新加入的方法 |
|---|---|
| Collection | removeIf()、spliterator()、 stream()、 parallelStream() 、forEach() |
| List | replaceAll() 、sort() |
| Map | getOrDefault()、 forEach()、 replaceAll()、 putIfAbsent() 、remove()、 replace() 、computeIfAbsent() 、computeIfPresent()、 compute()、 merge() |
这些新加入的方法大部分要用到java.util.function包下的接口,这意味着这些方法大部分都跟Lambda表达式相关。
相比Collection,Map中加入了更多的方法,下面一起了解一下。
(1)forEach() 以hashMap为例说明forEach()方法
该方法签名为void forEach(BiConsumer<? super K,? super V> action),作用是对Map中的每个映射执行action指定的操作,其中BiConsumer是一个函数接口,里面有一个待实现方法void accept(T t, U u)。BinConsumer接口名字和accept()方法名字都不重要,请不要记忆他们。
需求:假设有一个数字到对应英文单词的Map,请输出Map中的所有映射关系。
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
// Java7以及之前写法
for (Map.Entry<Integer, String> entry : map.entrySet()) {
System.out.println(entry.getKey() + "=" + entry.getValue());
}
// 使用Map.forEach()方法,并使用匿名内部类实现BiConsumer接口
map.forEach(new BiConsumer<Integer, String>() {
@Override
public void accept(Integer integer, String s) {
System.out.println("key=" + integer + " value=" + s);
}
});
// 使用lambda表达式
map.forEach((k,v)-> System.out.println("key="+k+" value="+v));
(2)getOrDefault() 以hashMap为例说明getOrDefault()方法
该方法跟Lambda表达式没关系,但是很有用。方法签名为V getOrDefault(Object key, V defaultValue),作用是按照给定的key查询Map中对应的value,如果没有找到则返回设置的默认值defaultValue。使用该方法程序员可以省去查询指定键值是否存在的麻烦。
需求;假设有一个数字到对应英文单词的Map,输出4对应的英文单词,如果不存在则输出NoValue。
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
// Java7以及之前做法
if (map.containsKey(6)) {
System.out.println(map.get(4));
} else {
System.out.println("NoValue");
}
// Java8使用Map.getOrDefault(),如果不存在直接返回NoValue
System.out.println(map.getOrDefault(6, "NoValue"));
(3)putIfAbsent() 以hashMap为例说明putIfAbsent()方法
该方法跟Lambda表达式没关系,但是很有用。方法签名为V putIfAbsent(K key, V value),作用是只有在不存在key值的映射或映射值为null时,才将value指定的值放入到Map中,否则不对Map做更改。该方法将条件判断和赋值合二为一,使用起来更加方便。
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
map.put(6,null);
map.putIfAbsent(6, "null-six");
map.forEach((k, v) -> {
System.out.println("k=" + k + " v=" + v);
});
(4)remove(Object key, Object value) 以hashMap为例说明remove()方法
我们都知道Map中有一个remove(Object key)方法,来根据指定key值删除Map中的映射关系;Java8新增了remove(Object key, Object value)方法,只有在当前Map中key正好映射到value时才删除该映射,否则什么也不做。
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
map.remove(5,"five");
map.forEach((k,v)-> System.out.println("key="+k+" value="+v));
(5)replace() 以hashMap为例说明replace()方法
在Java7及以前,要想替换Map中的映射关系可通过put(K key, V value)方法实现,该方法总是会用新值替换原来的值。为了更精确的控制替换行为,Java8在Map中加入了两个replace()方法,分别如下:
replace(K key, V value),只有在当前Map中key的映射存在时才用value去替换原来的值,否则什么也不做。replace(K key, V oldValue, V newValue),只有在当前Map中key的映射存在且等于oldValue时才用newValue去替换原来的值,否则什么也不做
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
// 因为map中key不可重复,因此会替换掉以前key对应的value
map.put(5, "five1");
// 如果存在对应的key,则替换掉对应key的值
map.replace(5, "six");
// map中存在key-value的映射才使用newValue替换掉oldValue
map.replace(5, "five", "newfive");
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
(6)replaceAll() 以hashMap为例说明replaceAll()方法
该方法签名为replaceAll(BiFunction<? super K,? super V,? extends V> function),作用是对Map中的每个映射执行function指定的操作,并用function的执行结果替换原来的value。其中BiFunction是一个函数接口,里面有一个待实现方法R apply(T t, U u)。
需求:假设有一个数字到对应英文单词的Map,请将原来映射关系中的单词都转换成大写。
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
// Java7以及之前替换Map中所有映射关系
for (Map.Entry<Integer, String> entry : map.entrySet()) {
entry.setValue(entry.getValue().toUpperCase());
}
// 遍历输出映射key-value
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
// 调用replaceAll()方法,并使用匿名内部类实现BiFunction接口
map.replaceAll(new BiFunction<Integer, String, String>() {
@Override
public String apply(Integer integer, String s) {
return s.toUpperCase();
}
});
// 遍历输出映射key-value
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
// 使用replaceAll()并结合Lambda表达式实现
map.replaceAll((k, v) -> v.toUpperCase());
// 遍历输出映射key-value
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
(7)merge() 以hashMap为例说明merge()方法
该方法签名为merge(K key, V value, BiFunction<? super V,? super V,? extends V> remappingFunction),作用是:如果Map中key对应的映射不存在或者为null,则将value(不能是null)关联到key上,否则执行remappingFunction;如果执行结果非null则用该结果跟key关联,否则在Map中删除key的映射。
也就是说如果key存在就把newvalue拼接在这个key对应的value上,如果不存在就把这个newValue与key进行映射,put到map中。如下代码所示:
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
map.put(6, "six");
// 使用匿名内部类实现BiFunction接口
map.merge(6, "+", new BiFunction<String, String, String>() {
@Override
public String apply(String s, String s2) {
return s + s2;
}
});
// 遍历输出key-value
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
// 使用lambda表达式实现
map.merge(6, "+", (s, s2) -> s + s2);
// 遍历输出key-value
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
(8)compute() 以hashMap为例说明compute()方法
该方法签名为compute(K key, BiFunction<? super K,? super V,? extends V> remappingFunction),作用是把remappingFunction的计算结果关联到key上,如果计算结果为null,则在Map中删除key的映射。 也就是说,如果计算的结果为null则把这个key-value映射给删掉,如果计算结果不为空,则把这个计算结果覆盖掉以前的value。
Map<Integer, String> map = new HashMap<>(16);
map.put(1, "one");
map.put(2, "two");
map.put(3, "three");
map.put(4, "four");
map.put(5, "five");
map.put(6, "six");
// 使用匿名内部类实现BiFunction接口写法
map.compute(5, new BiFunction<Integer, String, String>() {
@Override
public String apply(Integer integer, String s) {
return s == null ? "null" : s + " is not null";
}
});
// lambda表达式写法
map.compute(5, (integer, s) -> s == null ? "null" : s + "is not null");
// 遍历输出key-value映射
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
(9)computeIfAbsent() 以hashMap为例说明computeIfAbsent()方法
该方法签名为V computeIfAbsent(K key, Function<? super K,? extends V> mappingFunction),作用是:只有在当前Map中不存在key值的映射或映射值为null时,才调用mappingFunction,并在mappingFunction执行结果非null时,将结果跟key关联。
Function是一个函数接口,里面有一个待实现方法R apply(T t)。computeIfAbsent()常用来对Map的某个key值建立初始化映射。
比如我们要实现一个多值映射,Map的定义可能是Map<K,Set<V>>,要向Map中放入新值,可通过如下代码实现:
// 实现一个key对应多个值
Map<Integer, Set<String>> map = new HashMap<>(16);
// Java7及以前的实现方式
if (map.containsKey(1)) {
map.get(1).add("one");
} else {
Set<String> valueSet = new HashSet<>();
valueSet.add("one");
map.put(1, valueSet);
}
// Java8的实现方式 即它会判断一下这个key是否存在并且key对应的value是否为空,
// 如果key存在且key对应的value不为null,则将这个value关联到对应的key上,即在原来的value中新增一个value
// 如果key不存在,则新增一个key-value映射关系
map.computeIfAbsent(1, v -> new HashSet<>()).add("oneone");
// 遍历输出key-value映射
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
使用computeIfAbsent()将条件判断和添加操作合二为一,使代码更加简洁。
(10)computeIfPresent() 以hashMap为例说明computeIfPresent()方法
该方法签名为V computeIfPresent(K key, BiFunction<? super K,? super V,? extends V> remappingFunction),作用跟computeIfAbsent()相反,即只有在当前Map中存在key值的映射且非null时,才调用remappingFunction,如果remappingFunction执行结果为null,则删除key的映射,否则使用该结果替换key原来的映射。
这个函数的功能跟如下代码是等效的:
Map<Integer, Set<String>> map = new HashMap<>(16);
// Java7及以前的实现方式
if (map.containsKey(1)) {
map.get(1).add("one");
} else {
Set<String> valueSet = new HashSet<>();
valueSet.add("one");
map.put(1, valueSet);
}
// 匿名内部类实现BiFunction接口,如果key存在并且计算结果不为null时将计算的结果替换掉key对应的原来的值
map.computeIfPresent(8, new BiFunction<Integer, Set<String>, Set<String>>() {
@Override
public Set<String> apply(Integer integer, Set<String> strings) {
Set<String> set = new HashSet<>();
set.add("888");
return set;
}
});
// lambda表达式实现
map.computeIfPresent(8, (integer, strings) -> {
Set<String> set = new HashSet<>();
set.add("888");
return set;
});
// 遍历输出key-value映射
map.forEach((k, v) -> System.out.println("key=" + k + " value=" + v));
在使用lambda表达式时需要明白以下两点:
- Java8为容器新增一些有用的方法,这些方法有些是为完善原有功能,有些是为引入函数式编程,学习和使用这些方法有助于我们写出更加简洁有效的代码。
- 函数接口虽然很多,但绝大多数时候我们根本不需要知道它们的名字,书写Lambda表达式时类型推断帮我们做了一切。
参考博文:
(1)https://objcoding.com/2019/03/04/lambda/ (非常详细,值得仔细阅读)
(2)https://www.runoob.com/java/java8-lambda-expressions.html