Java 8 - 如何将属性值从一个列表复制到另一个列表?

时间:2015-11-26 12:09:46

标签: java lambda java-8

我有两个List,例如:

List<Foo> list1 = Lists.newArrayList(new Foo(...),...);
List<Bar> list2 = Lists.newArrayList(new Bar(...),...);

Bar中有一个属性fooId。假设list1.size() == list2.size()。 我想按顺序设置fooId个实例的Bar个。

我尝试了以下代码:

int index = 0;
list2.forEach(b -> b.fooId = list1.get(index++).getId());

但编译失败

  

在封闭范围内定义的局部变量索引必须是最终的或有效的最终

Java 8是否有一些方便的方法来处理它?<​​/ p>

4 个答案:

答案 0 :(得分:4)

您无法修改lambda表达式中的index变量。

您可以使用IntStream迭代索引(尽管与简单的for循环相比,这并没有太大的改进):

IntStream.range(0,list1.size())
         .forEach(i -> list2.get(i).setFooId(list1.get(i).getId()));

答案 1 :(得分:3)

你可以这样做:

~/Downloads ❯ ffmpeg -i working.mp3 -i not_working.mp3
ffmpeg version 2.8.3 Copyright (c) 2000-2015 the FFmpeg developers
  built with Apple LLVM version 7.0.0 (clang-700.1.76)
  configuration: --prefix=/usr/local/Cellar/ffmpeg/2.8.3 --enable-shared --enable-pthreads --enable-gpl --enable-version3 --enable-hardcoded-tables --enable-avresample --cc=clang --host-cflags= --host-ldflags= --enable-opencl --enable-libx264 --enable-libmp3lame --enable-libvo-aacenc --enable-libxvid --enable-vda
  libavutil      54. 31.100 / 54. 31.100
  libavcodec     56. 60.100 / 56. 60.100
  libavformat    56. 40.101 / 56. 40.101
  libavdevice    56.  4.100 / 56.  4.100
  libavfilter     5. 40.101 /  5. 40.101
  libavresample   2.  1.  0 /  2.  1.  0
  libswscale      3.  1.101 /  3.  1.101
  libswresample   1.  2.101 /  1.  2.101
  libpostproc    53.  3.100 / 53.  3.100
[mp3 @ 0x7fd2d380da00] Skipping 0 bytes of junk at 33.
[mp3 @ 0x7fd2d380da00] Estimating duration from bitrate, this may be inaccurate
Input #0, mp3, from 'working.mp3':
  Metadata:
    encoder         : Lavf52.64.2
  Duration: 00:00:00.65, start: 0.000000, bitrate: 64 kb/s
    Stream #0:0: Audio: mp3, 22050 Hz, mono, s16p, 64 kb/s
[mp3 @ 0x7fd2d4008800] Skipping 0 bytes of junk at 417.
Input #1, mp3, from 'not_working.mp3':
  Duration: 00:00:01.83, start: 0.025057, bitrate: 46 kb/s
    Stream #1:0: Audio: mp3, 44100 Hz, stereo, s16p, 46 kb/s
    Metadata:
      encoder         : LAME3.99r

但请注意,这不是很好的功能代码。你最好写一个明确的IntStream.range(0, list1.size()) .forEach(i -> list2.get(i).fooId = list1.get(i).getId()); 循环,而不是使用Stream。

您问题中的代码不起作用,因为您无法在lambda表达式中引用非最终(或有效最终)的变量。

如果您在此操作后不介意for并且可以删除其元素,您也可以使用此功能:

list1

答案 2 :(得分:2)

不幸的是,Java 8流API中没有zip操作。但您可以使用protonpack库。

List<Bar> upd = StreamUtils.zip(list1.stream(), list2.stream(), (foo, bar) -> {
  bar.setId(foo.getId());
  return bar;
}).collect(Collectors.toList());

我还想补充说,使用可变数据对象是一种必要的风格。在函数式中,如果需要更改任何内容,最好使用不可变数据并返回新对象。

答案 3 :(得分:2)

使用我的StreamEx库,您可以减少样板:

EntryStream.zip(list1, list2).forKeyValue((foo, bar) -> bar.fooId = foo.getId());

EntryStream.zip为每对Map.Entry<Foo, Bar>list1创建一个list2的流。 forKeyValue方法接受BiConsumer(它是forEach的语法糖)。在内部,它接近@Eran解决方案。