java lists.transform_Guava Lists.transform 使用及问题

Guava的Lists.transform方法在转换列表时可能会带来意想不到的效果,当源列表被修改时,转换后的列表也会同步变化。本文通过示例展示了这个问题,并探讨了Lists.transform的实现细节,提醒开发者在使用时需谨慎,避免潜在问题。
摘要由CSDN通过智能技术生成

简介:Guava 提供的Lists.transform很强大,但在项目中最近的使用遇到了问题,发现还是有坑的,所以本文先是简单介绍Lists.transform和其使用,后面结合实际的使用过程可能出现的问题来一起看下Lists.transform的实现原理,如有不对的地方欢迎评论讨论。

一:Lists.transform的使用

大家在写代码的过程中肯定会碰到一种状况,dao中查询数据库返回了一个结果集list,其中Result对象中包含了Id,nameStr,msg等字段,但是上层业务的一些接口参数可能只需要id的结果集list,当然我们可以使用for each循环list然后将Id取出依次add到list中,Lists.transform就是帮我们方便的解决这一过程,能够轻松的从一种类型的list转换为另一种类型的list。

使用方式如下:

public void listToList(){

//源list

List listResults = Lists.newArrayList(new Result(1,"test1"),new Result(2,"test2"),new Result(3,"test3"));

//转换为目标list

List strLists = Lists.transform(listResults,new Function(){

@Override

public String apply(Result result){

return result.getNameStr();

}

});

}

二:Lists.transform使用可能遇到的问题

请看如下代码,只是在上面的示例中增加了一行改变listResults中对象属性的操作和打印:

@Test

public void listToList(){

//源list

List listResults = Lists.newArrayList(new Result(1,"test1"),new Result(2,"test2"),new Result(3,"test3"));

//转换为目标list

List strLists = Lists.transform(listResults,new Function(){

@Override

public String apply(Result result){

return result.getNameStr();

}

});

//转换后目标list打印

System.out.println("strLists 1 values:");

for(String str:strLists){

System.out.println(str+";");

}

//修改源list的值

for(Result result:listResults){

result.setNameStr("reset");

}

//再次打印目标list

System.out.println("strLists 2 values:");

for(String str:strLists){

System.out.println(str+";");

}

}

输出结果是什么呢?如果刚开始使用Lists.transform的话,肯定会认为两次输出是相同的才对,但实际情况却不是这样的,已经转换后得到的list会受到源list的改动而改变,上面代码的输出结果如下:

strLists 1 values:

test1;

test2;

test3;

strLists 2 values:

reset;

reset;

reset;

这个坑如果不了解就使用Lists.transform的话那么问题有点大了,会造成很严重的问题,我们先来看下Lists.transform是怎样实现的。

public static List transform(List fromList, Function super F, ? extends T> function) {

return (List)(fromList instanceof RandomAccess ? new Lists.TransformingRandomAccessList(fromList, function) : new Lists.TransformingSequentialList(fromList, function));

}

可以看到上面的代码写的很清楚,Lists.transform返回的是一个新创建的TransformingSequentialList实例,然后我们再接着往下看

public ListIterator listIterator(int index) {

return new TransformedListIterator(this.fromList.listIterator(index)) {

T transform(F from) {

return TransformingSequentialList.this.function.apply(from);

}

};

}

TransformingSequentialList每次遍历都会从原来的list中遍历来从新计算得到function

总结

使用新的开源类库时最后多做了解,看下源码后再使用,这样就能在合适的场景下合理使用,如果不了解的话还是最好先用比较稳妥的方式进行编码。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值