往期热门文章:
1、多线程使用不当导致的 OOM
2、老板要我开发一个简单的工作流引擎
3、Spring Boot 启动时自动执行代码的几种方式,还有谁不会??
4、Lombok原理和同时使⽤@Data和@Builder 的坑
4、如何用 Java 几分钟处理完 30 亿个数据?
背景
google提供的Lists.transform⽅法可以⽤来将⼀个实体类的List通过⼀定的逻辑转化为另外⼀个实体类的List。例如下⾯的示例,将⼀个string的list转换为integer的list。
举个例子:List A 转化为 ListB。如下
public static void main(String[] args) {List tests = Stream.of("123", "321").collect(Collectors.toList());List integers = Lists.transform(tests, t -> {
return t;});
}
问题
集合ids在Lists.transform⽅法add一个值,但是最后并没有add进去。
public static void main(String[] args) {List ids&#61;new ArrayList<>();List tests &#61; Stream.of("123", "321").collect(Collectors.toList());List integers &#61; Lists.transform(tests, t -> {ids.add(1L);
return Integer.parseInt(t);});System.out.println("结果&#xff1a;"&#43;JSON.toJSON(ids));
}
打印结果如下所示&#xff1a;
我们打上断点看下是否执行
可以看到第一个断点没有进去&#xff0c;直接走的第二个断点。
原因从上述现象可以看出
Lists.transform⽅法实际上不会去⽴刻做转换操作&#xff0c;⽽是延迟转化。
查询源码得知,我们的 sourceList 在执⾏ Lists.transform 后是没有⽴刻被赋值的&#xff0c;它只是被转换成了TransformingRandomAccessList 类型&#xff0c;此类型 重写了迭代器&#xff0c;每次当我们迭代 userFormList时候&#xff0c;它才会调⽤ function 进⾏赋值。
⽽且每次在foreach的时候传⼊的function都会执⾏⼀遍&#xff0c;需要function内的⽅法⽐较有效率。因此使⽤不当&#xff0c;会有问题。原理有点像sublist&#xff0c;他实际上返回的是⼀个内部类。
解决方式第一种方式&#xff1a;使用包装类包装返回的内部对象
public static void main(String[] args) {List ids &#61; new ArrayList<>();List tests &#61; Stream.of("123", "321").collect(Collectors.toList());
// List integers &#61; Lists.transform(tests, t -> {
// ids.add(1L);
// return Integer.parseInt(t);
// });List integers &#61; new ArrayList<>(Lists.transform(tests, t -> {ids.add(1L);
return Integer.parseInt(t);}));System.out.println("结果&#xff1a;" &#43; JSON.toJSON(ids));}
执行结果&#xff1a;
第二种方式&#xff1a;实现⼀个简单的transform不⽤google提供的⽅法
public static List transformList(List fromList, Function fuction) {
if (fromList &#61;&#61; null) {
return new ArrayList<>();}
List lists &#61; new ArrayList<>();
for (F from : fromList) {lists.add(fuction.apply(from));}
return lists;
}
写一个main方法来测试用法&#xff1a;
public static void main(String[] args) {List ids &#61; new ArrayList<>();List tests &#61; Stream.of("123", "321").collect(Collectors.toList());
// List integers &#61; Lists.transform(tests, t -> {
// ids.add(1L);
// return Integer.parseInt(t);
// });
// List integers &#61; new ArrayList<>(Lists.transform(tests, t -> {
// ids.add(1L);
// return Integer.parseInt(t);
// }));List tet2List&#61;new ArrayList<>();Tet2 tet2&#61;new Tet2(1L,"haha");tet2List.add(tet2);List tet1List &#61; transformList(tet2List, m -> {Tet1 tet1 &#61; new Tet1();tet1.setId(m.getId());tet1.setName(m.getName());
return tet1;});System.out.println("结果&#xff1a;" &#43; JSON.toJSON(tet1List));}
打印结果如下&#xff1a;
总结使⽤第三⽅开源框架的时候&#xff0c;务必了解其原理&#xff0c;防⽌第三⽅框架使⽤过程中隐藏的坑。
转自&#xff1a;树洞君
链接&#xff1a;https://juejin.cn/post/7105581965503365127
最近热文阅读&#xff1a;
1、面试官 | Spring Boot 项目如何统一结果&#xff0c;统一异常&#xff0c;统一日志&#xff1f;
2、为什么不建议使用ON DUPLICATE KEY UPDATE&#xff1f;
3、Java8 Stream&#xff0c;过分丝滑&#xff01;
4、8 种最坑SQL语法&#xff0c;工作中踩过吗&#xff1f;
5、Java 语言“坑爹” TOP 10
6、你还不明白如何解决分布式Session&#xff1f;看这篇就够了&#xff01;
7、能解决 80% 故障的排查思路
8、程序员坐牢了&#xff0c;会被安排写代码吗&#xff1f;
9、面试被问Nginx&#xff0c;怎么破?
10、为什么很多 SpringBoot 开发者放弃了 Tomcat&#xff0c;选择了 Undertow?
关注公众号&#xff0c;你想要的Java都在这里