Lists.transform() ⽅法使⽤的坑

>>强大,10k+点赞的 SpringBoot 后台管理系统竟然出了详细教程!

点击关注公众号,实用技术文章及时了解Lists.transform() ⽅法使⽤的坑

背景

google提供的Lists.transform⽅法可以⽤来将⼀个实体类的List通过⼀定的逻辑转化为另外⼀个实体类的List。例如下⾯的示例,将⼀个string的list转换为integer的list。

举个例子:List< String> A 转化为 List< Integer>B。如下

public static void main(String[] args) {
    List<String> tests = Stream.of("123""321").collect(Collectors.toList());
    List<Integer> integers = Lists.transform(tests, t -> {
        return t;
    });
}

问题

集合ids在Lists.transform⽅法add一个值,但是最后并没有add进去。

public static void main(String[] args) {
    List<Long> ids=new ArrayList<>();
    List<Integer> tests = Stream.of("123""321").collect(Collectors.toList());
    List<Integer> integers = Lists.transform(tests, t -> {
        ids.add(1L);
        return Integer.parseInt(t);
     });
    System.out.println("结果:"+JSON.toJSON(ids));
}

打印结果如下所示:

Lists.transform() ⽅法使⽤的坑

我们打上断点看下是否执行

可以看到第一个断点没有进去,直接走的第二个断点。

Lists.transform() ⽅法使⽤的坑

原因

从上述现象可以看出

  • Lists.transform⽅法实际上不会去⽴刻做转换操作,⽽是延迟转化。

  • 查询源码得知,我们的 sourceList 在执⾏ Lists.transform 后是没有⽴刻被赋值的,它只是被转换成了TransformingRandomAccessList 类型,此类型 重写了迭代器,每次当我们迭代 userFormList时候,它才会调⽤ function 进⾏赋值。

  • ⽽且每次在foreach的时候传⼊的function都会执⾏⼀遍,需要function内的⽅法⽐较有效率。因此使⽤不当,会有问题。原理有点像sublist,他实际上返回的是⼀个内部类。

Lists.transform() ⽅法使⽤的坑
Lists.transform() ⽅法使⽤的坑

解决方式

第一种方式:使用包装类包装返回的内部对象

public static void main(String[] args) {
        List<Long> ids = new ArrayList<>();
        List<String> tests = Stream.of("123""321").collect(Collectors.toList());
//        List<Integer> integers = Lists.transform(tests, t -> {
//            ids.add(1L);
//            return Integer.parseInt(t);
//        });
        List<Integer> integers = new ArrayList<>(Lists.transform(tests, t -> {
            ids.add(1L);
            return Integer.parseInt(t);
        }));

        System.out.println("结果:" + JSON.toJSON(ids));
    }

执行结果:

Lists.transform() ⽅法使⽤的坑

第二种方式:实现⼀个简单的transform不⽤google提供的⽅法

public static <F, T> List<T> transformList(List<F> fromList, Function<F, T> fuction) {
    if (fromList == null) {
        return new ArrayList<>();
    }
    List<T> lists = new ArrayList<>();
    for (F from : fromList) {
        lists.add(fuction.apply(from));
    }
    return lists;
}

写一个main方法来测试用法:

public static void main(String[] args) {
        List<Long> ids = new ArrayList<>();
        List<String> tests = Stream.of("123""321").collect(Collectors.toList());
//        List<Integer> integers = Lists.transform(tests, t -> {
//            ids.add(1L);
//            return Integer.parseInt(t);
//        });
//        List<Integer> integers = new ArrayList<>(Lists.transform(tests, t -> {
//            ids.add(1L);
//            return Integer.parseInt(t);
//        }));
        List<Tet2> tet2List=new ArrayList<>();
        Tet2 tet2=new Tet2(1L,"haha");
        tet2List.add(tet2);
        List<Tet1> tet1List = transformList(tet2List, m -> {
            Tet1 tet1 = new Tet1();
            tet1.setId(m.getId());
            tet1.setName(m.getName());
            return tet1;
        });
        System.out.println("结果:" + JSON.toJSON(tet1List));
    }

打印结果如下:

Lists.transform() ⽅法使⽤的坑

总结

使⽤第三⽅开源框架的时候,务必了解其原理,防⽌第三⽅框架使⽤过程中隐藏的坑。

来源:juejin.cn/post/7105581965503365127

推荐

Java面试题宝典

技术内卷群,一起来学习!!

Lists.transform() ⽅法使⽤的坑

PS:因为公众号平台更改了推送规则,如果不想错过内容,记得读完点一下“在看”,加个“星标”,这样每次新文章推送才会第一时间出现在你的订阅列表里。“在看”支持我们吧!

原文始发于微信公众号(Java知音):Lists.transform() ⽅法使⽤的坑