美文网首页
steam map问题

steam map问题

作者: 那一半橘子 | 来源:发表于2020-03-12 15:05 被阅读0次

    NullPointerException

    v 为null,报空指针错误

    k为null, 运行正常。

    以往的认知:HashMap中k,v都是可以存null值的。在上面的测试用例中可以看到,v为null其实会报错。

    查看API可以看到

    publicstatic<T,K,U,MextendsMap<K,U>>Collector<T,?,M>toMap(Function<?superT,?extendsK>keyMapper,Function<?superT,?extendsU>valueMapper,BinaryOperator<U>mergeFunction,Supplier<M>mapSupplier){BiConsumer<M,T>accumulator=(map,element)->map.merge(keyMapper.apply(element),valueMapper.apply(element),mergeFunction);returnnewCollectorImpl<>(mapSupplier,accumulator,mapMerger(mergeFunction),CH_ID);}

    @OverridepublicVmerge(Kkey,Vvalue,BiFunction<?super V,?super V,?extends V>remappingFunction){if(value==null)thrownewNullPointerException();if(remappingFunction==null)thrownewNullPointerException();inthash=hash(key);Node<K,V>[]tab;Node<K,V>first;intn,i;intbinCount=0;TreeNode<K,V>t=null;Node<K,V>old=null;if(size>threshold||(tab=table)==null||(n=tab.length)==0)n=(tab=resize()).length;if((first=tab[i=(n-1)&hash])!=null){if(first instanceof TreeNode)old=(t=(TreeNode<K,V>)first).getTreeNode(hash,key);else{Node<K,V>e=first;Kk;do{if(e.hash==hash&&((k=e.key)==key||(key!=null&&key.equals(k)))){old=e;break;}++binCount;}while((e=e.next)!=null);}}if(old!=null){Vv;if(old.value!=null)v=remappingFunction.apply(old.value,value);elsev=value;if(v!=null){old.value=v;afterNodeAccess(old);}elseremoveNode(hash,key,null,false,true);returnv;}if(value!=null){if(t!=null)t.putTreeVal(this,tab,hash,key,value);else{tab[i]=newNode(hash,key,value,first);if(binCount>=TREEIFY_THRESHOLD-1)treeifyBin(tab,hash);}++modCount;++size;afterNodeInsertion(true);}returnvalue;}

    从代码中可以看到,HashMap.merge 会对v进行判空。

    目前我的解决办法是在流中加上判空过滤

    System.out.println(list.stream().filter(t->t.getName()!=null).collect(Collectors.toMap(TestDO::getId,TestDO::getName,(k1,k2)->k1)));

    Duplicate key

    查看toMap的API及相关注释

    /*

    * <p>If the mapped keys contains duplicates (according to

        * {@link Object#equals(Object)}), an {@code IllegalStateException} is

        * thrown when the collection operation is performed.  If the mapped keys

        * may have duplicates, use {@link #toMap(Function, Function, BinaryOperator)}

        * instead.

    */publicstatic<T,K,U>Collector<T,?,Map<K,U>>toMap(Function<?superT,?extendsK>keyMapper,Function<?superT,?extendsU>valueMapper){returntoMap(keyMapper,valueMapper,throwingMerger(),HashMap::new);}

    如果有重复的key,使用toMap(Function, Function)会抛异常,可以使用toMap(Function, Function, BinaryOperator)来解决。

    优化后的代码是

    publicclassMailTest{publicstaticvoidmain(String[]args){List<TestDO>list=Lists.newArrayList();TestDO t1=newTestDO(1,"a");TestDO t2=newTestDO(1,"b");list.add(t1);list.add(t2);System.out.println(list.stream().collect(Collectors.toMap(TestDO::getId,TestDO::getName,(k1,k2)->k1)));System.out.println(list.stream().collect(Collectors.toMap(TestDO::getId,TestDO::getName,(k1,k2)->k2)));System.out.println(list.stream().collect(Collectors.toMap(TestDO::getId,TestDO::getName,(k1,k2)->k1+k2)));}}

    从执行结果来看,对于重复的key,对他的value可以有三种取值情况:

    (k1, k2) -> k1,取第一个值

    (k1, k2) -> k2),取第二个值

    (k1, k2) -> k1 + k2),对两个值做某些操作后取结果。(例如基本类型可以进行加减乘除)。

    作者:snoweek

    链接:https://www.jianshu.com/p/6a37e245ff46

    来源:简书

    著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

    相关文章

      网友评论

          本文标题:steam map问题

          本文链接:https://www.haomeiwen.com/subject/wmtpjhtx.html