背景
很多时候我们需要用到有序的Map,需要使用LinkedHashMap。如果我们需要对LinkedHashMap做一个过滤呢?是不是可以使用Stream的Filter?看起来很简单的一个问题,伴随展开却能反映出很多。我先给出结论,然后使用一个LinkedHashMap过滤值中包含1的小需求来展开不同的思考。
先给出结论:
- 不要书写团队人员都觉得晦涩的代码
- 不同业务代码中共用的业务无关的技术重复代码应该抽取
- Collectors.toMap的默认收集结果并不是LinkedHashMap
错误写法-直接使用Stream的Filter
正常的Stream加Filter写法如下:使用普通的MapStream来过滤,然后使用Collectors.toMap,结果是一个无序的Map。
Map<String, String> map = new LinkedHashMap<>();
map.put("a", "a1");
map.put("aa", "a2");
map.put("b", "b1");
map.put("bb", "b2");
System.out.println(map);
//筛选出来包含1的,并保持有序
map = map.entrySet().stream()
.filter(e -> e.getValue().contains("1"))
.collect(Collectors.toMap(e -> e.getKey(), e -> e.getValue()));
System.out.println(map);
复制代码
我们想要的是LinkedHashMap,是否可以强制类型转换为LinkedHashMap呢?
LinkedHashMap<String, String> map = new LinkedHashMap<>();
map.put("a", "a1");
map.put("aa", "a2");
map.put("b", "b1");
map.put("bb", "b2");
System.out.println(map);
map = (LinkedHashMap) map.entrySet().stream()
.filter(e -> e.getValue().contains("1"))
.collect(Collectors.toMap(e -> e.getKey(), e -> e.getValue()));
System.out.println(map);
复制代码
执行结果会报错:
{a=a1, aa=a2, b=b1, bb=b2}
Exception in thread "main" java.lang.ClassCastException: class java.util.HashMap cannot be cast to class java.util.LinkedHashMap (java.util.HashMap and java.util.LinkedHashMap are in module java.base of loader 'bootstrap')
at fly.sample.collection.LinkedHashMapFilterTest.test2(LinkedHashMapFilterTest.java:40)
at fly.sample.collection.LinkedHashMapFilterTest.main(LinkedHashMapFilterTest.java:10)
复制代码
原因是因为toMap的结果不是原来的类型LinkedHashMap,而是新建立的HashMap,看下toMap的源码
public static <T, K, U>
Collector<T, ?, Map<K,U>> toMap(Function<? super T, ? extends K> keyMapper,
Function<? super T, ? extends U> valueMapper) {
return new CollectorImpl<>(HashMap::new,
uniqKeysMapAccumulator(keyMapper, valueMapper),
uniqKeysMapMerger(),
CH_ID);
}
复制代码
Sream的正确写法
LinkedHashMap<String, String> map = new LinkedHashMap<>();
map.put("a", "a1");
map.put("aa", "a2");
map.put("b", "b1");
map.put("bb", "b2");
System.out.println(map)
map = map.entrySet()
.stream().filter(e -> e.getValue().contains("1")).collect(Collectors.toMap(e -> e.getKey(), v -> v.getValue(),
(oldValue, newValue) -> oldValue, LinkedHashMap::new));
System.out.println("4" + map);
复制代码
可以正常执行,输出依然有序
{a=a1, aa=a2, b=b1, bb=b2}
4{a=a1, b=b1}
复制代码
正确的写法易读性很差,很多人都会抱怨干嘛不用原始写法。所以个人建议自己抽取个可读性更好的方法,这样不仅仅可读性好,不同业务之间过滤LinkedHashMap可以共用代码。
通用方法抽取
第一种通用写法呼应下不喜欢晦涩写法的一部分人,可以抽出如下方法:
public static <K, V> LinkedHashMap<K, V> filterLinkedHashMap(LinkedHashMap<K, V> linkedHashMap, Predicate<Map.Entry<K, V>> predicate) {
LinkedHashMap resultMap = new LinkedHashMap<>();
for (Map.Entry entry : linkedHashMap.entrySet()) {
if (predicate.test(entry)) {
resultMap.put(entry.getKey(), entry.getValue());
}
}
return resultMap;
}
复制代码
程序中只要如下使用filterLinkedHashMap方法就可以,易读性就可以得到极大提高。
LinkedHashMap<String, String> map = new LinkedHashMap<>();
map.put("a", "a1");
map.put("aa", "a2");
map.put("b", "b1");
map.put("bb", "b2");
System.out.println(map)
map = filterLinkedHashMap(map, e -> e.getValue().contains("1"));
System.out.println(map);
复制代码
当然,你也可以选择filterLinkedHashMap的实现使用晦涩的写法,因为对于方法使用者是不可见的。