如何将Java 8 Stream转换为数组?


775

将Java 8 Stream转换为数组的最简单/最快捷的方法是什么?


2
我建议您还原回滚,因为问题比较完整,并表明您已尝试过一些操作。
skiwi 2014年

2
@skiwi谢谢!但我认为尝试执行的代码并未真正为问题添加更多信息,并且没有人尖叫“向我们展示您的尝试” =)

17
@skiwi:尽管我通常会大声地回答“我做家庭作业,而不是我做”的问题,但是这个问题对我来说似乎很清楚,没有任何其他混乱。让我们保持整洁。
Honza Zidek

:您可以在包的官方文档很多答案和指导docs.oracle.com/javase/8/docs/api/java/util/stream/...
克里斯托夫•鲁西

Answers:


1168

最简单的方法是将该toArray(IntFunction<A[]> generator)方法与数组构造函数引用一起使用。API文档中建议使用该方法

String[] stringArray = stringStream.toArray(String[]::new);

它的作用是找到一个将整数(大小)作为参数的方法,并返回一个 String[],而这正是(的重载之一)new String[]所做的。

您也可以编写自己的IntFunction

Stream<String> stringStream = ...;
String[] stringArray = stringStream.toArray(size -> new String[size]);

目的 IntFunction<A[]> generator是将整数(数组的大小)转换为新数组。

示例代码:

Stream<String> stringStream = Stream.of("a", "b", "c");
String[] stringArray = stringStream.toArray(size -> new String[size]);
Arrays.stream(stringArray).forEach(System.out::println);

印刷品:

a
b
c

8
这里是解释为何以及如何数组构造参考实际工作:stackoverflow.com/questions/29447561/...
jarek.jpa

“ Zenexer是正确的,解决方案应该是:stream.toArray(String [] :: new);” ...很好,但是应该理解方法引用在逻辑上和功能上都等效,toArray(sz -> new String[sz])因此我不确定是否真的可以说出解决方案应该或必须是什么。
斯科特,

3
@scottb sz -> new String[sz]会创建一个新函数,而构造函数引用则不会。我猜这取决于您对垃圾收集流失的重视程度。
WORMSS

3
@WORMSS不是。它(静态地!)创造了一个新的,private method,它不会引起用户流失,并且两个版本都需要创建一个新对象。引用创建一个对象,该对象直接指向目标方法。Lambda创建一个指向所生成对象的对象private。由于缺少间接性且更易于VM优化,因此对构造函数的引用仍应具有更好的性能,但搅动与之无关。
HTNW

2
@HTNW,您是对的,对不起。实际上,调试的尝试实际上是造成搅动的原因,而这是我第一次尝试进行搅动的原因,因此,我一直将这种想法卡在脑海中。(发生这种情况时请讨厌它)。
WORMSS

41

如果要从Stream获取一个值数组,其值为1到10,则可以使用IntStream。

在这里,我们使用Stream.of方法创建一个Stream,并使用mapToInt将Stream转换为IntStream。然后我们可以调用IntStream的toArray方法。

Stream<Integer> stream = Stream.of(1,2,3,4,5,6,7,8,9,10);
//or use this to create our stream 
//Stream<Integer> stream = IntStream.rangeClosed(1, 10).boxed();
int[] array =  stream.mapToInt(x -> x).toArray();

这是同一件事,没有Stream,仅使用IntStream

int[]array2 =  IntStream.rangeClosed(1, 10).toArray();

15

您可以使用以下简单代码块将Java 8流转换为数组:

 String[] myNewArray3 = myNewStream.toArray(String[]::new);

但是,让我们进一步解释一下,首先,我们创建一个包含三个值的字符串列表:

String[] stringList = {"Bachiri","Taoufiq","Abderrahman"};

从给定的Array创建一个流:

Stream<String> stringStream = Arrays.stream(stringList);

我们现在可以在此流Ex上执行一些操作:

Stream<String> myNewStream = stringStream.map(s -> s.toUpperCase());

最后使用以下方法将其转换为Java 8 Array:

1-经典方法(功能接口)

IntFunction<String[]> intFunction = new IntFunction<String[]>() {
    @Override
    public String[] apply(int value) {
        return new String[value];
    }
};


String[] myNewArray = myNewStream.toArray(intFunction);

2-Lambda表达式

 String[] myNewArray2 = myNewStream.toArray(value -> new String[value]);

3-方法参考

String[] myNewArray3 = myNewStream.toArray(String[]::new);

方法参考说明:

这是编写lambda表达式的另一种方法,它与另一方法完全等效。



4

您可以创建将流转换为数组的自定义收集器。

public static <T> Collector<T, ?, T[]> toArray( IntFunction<T[]> converter )
{
    return Collectors.collectingAndThen( 
                  Collectors.toList(), 
                  list ->list.toArray( converter.apply( list.size() ) ) );
}

和快速使用

List<String> input = Arrays.asList( ..... );

String[] result = input.stream().
         .collect( CustomCollectors.**toArray**( String[]::new ) );

5
为什么要使用它代替Stream.toArray(IntFunction)
Didier L

我需要一个收集器传递给2-arg,Collectors.groupingBy以便可以将每个属性值的某些属性映射到对象数组。这个答案给了我确切的答案。还有@DidierL。
Ole VV

3

使用 toArray(IntFunction<A[]> generator)方法确实是一种非常优雅且安全的方法,可以将一个Stream转换(或更正确地,将其收集)为相同类型的Stream的数组。

但是,如果返回的数组的类型不重要,则简单地使用该toArray()方法既简单简短。例如:

    Stream<Object> args = Stream.of(BigDecimal.ONE, "Two", 3);
    System.out.printf("%s, %s, %s!", args.toArray());

0
Stream<Integer> stream = Stream.of(1, 2, 3, 4, 5, 6);

int[] arr=   stream.mapToInt(x->x.intValue()).toArray();

0
import java.util.List;
import java.util.stream.Stream;

class Main {

    public static void main(String[] args) {
        // Create a stream of strings from list of strings
        Stream<String> myStreamOfStrings = List.of("lala", "foo", "bar").stream();

        // Convert stream to array by using toArray method
        String[] myArrayOfStrings = myStreamOfStrings.toArray(String[]::new);

        // Print results
        for (String string : myArrayOfStrings) {
            System.out.println(string);
        }
    }
}

在线尝试:https : //repl.it/@SmaMa/Stream-to-array


您的答案和接受的答案有什么区别?

@LongNguyen这是一个完整的示例,其中包括一个在线重播场景,而不仅仅是一个片段。
Sma Ma


-2

您可以通过几种方式来做到。所有方式在技术上都是相同的,但使用Lambda可以简化一些代码。假设我们首先使用String初始化List,将其称为person。

List<String> persons = new ArrayList<String>(){{add("a"); add("b"); add("c");}};
Stream<String> stream = persons.stream();

现在,您可以使用以下两种方式之一。

  1. 使用Lambda Expresiion创建具有定义大小的新StringArray。

    String [] stringArray = stream.toArray(size-> new String [size]);

  2. 直接使用方法参考。

    String [] stringArray = stream.toArray(String [] :: new);

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.