Преобразование ArrayList <String> в 'String []' в Java


1025

Как я могу преобразовать ArrayList<String>объект в String[]массив в Java?


2
Сделали этот ответ с обновленным подходом с JDK-11, представляющим новый не менее производительный API toArray(T[])и похожий по синтаксису Stream.toArray.
Наман

Ответы:


1792
List<String> list = ..;
String[] array = list.toArray(new String[0]);

Например:

List<String> list = new ArrayList<String>();
//add some stuff
list.add("android");
list.add("apple");
String[] stringArray = list.toArray(new String[0]);

toArray()Метод без передачи каких - либо аргументы возвращаются Object[]. Таким образом, вы должны передать массив в качестве аргумента, который будет заполнен данными из списка и возвращен. Вы также можете передать пустой массив, но вы также можете передать массив с желаемым размером.

Важное обновление : изначально код выше использовался new String[list.size()]. Тем не менее, этот пост показывает, что благодаря оптимизации JVM использование new String[0]стало лучше.


16
Размер аргумента имеет какое-либо значение?
Турбьерн Равн Андерсен

37
это сохраняет еще один экземпляр массива
Божо

59
Оказывается, что предоставление массива нулевой длины, даже его создание и удаление, в среднем быстрее, чем выделение массива правильного размера. Ориентиры и объяснения см. Здесь: shipilev.net/blog/2016/arrays-wisdom-ancients
Стюарт Маркс

2
(Извините. Риторический вопрос. Ответ, конечно, потому что Java не делает настоящие дженерики; она в основном просто подделывает их, используя Object)
Nyerguds

3
@lyuboslavkanev Проблема в том, что наличия универсального типа в Java недостаточно для создания объектов на основе этого типа; даже не массив (что смешно, потому что это должно работать для любого типа). Насколько я могу видеть, все, что можно сделать, это кастинг. На самом деле, даже для создания объектов типа вам необходим фактический Classобъект, который, по-видимому, совершенно невозможно извлечь из универсального типа. Причина этого, как я уже сказал, в том, что вся конструкция является поддельной; все это просто хранится как объект внутри.
Nyerguds

170

Альтернатива в Java 8:

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

Java 11+:

String[] strings = list.toArray(String[]::new);

25
Или какие-то преимущества?
Джеймс Уоткинс

1
Я бы предпочел этот синтаксис, но IntelliJ отображает ошибку компилятора, жалуется, что «T [] не является функциональным интерфейсом».
Глен Мазза

3
@GlenMazza вы можете использовать только toArrayна Streamобъекте. Эта ошибка компиляции может возникнуть, если вы уменьшите поток с помощью, Collectorsа затем попытаетесь применить toArray.
Удара Бентота

39

Вы можете использовать toArray()метод для List:

ArrayList<String> list = new ArrayList<String>();

list.add("apple");
list.add("banana");

String[] array = list.toArray(new String[list.size()]);

Или вы можете вручную добавить элементы в массив:

ArrayList<String> list = new ArrayList<String>();

list.add("apple");
list.add("banana");

String[] array = new String[list.size()];

for (int i = 0; i < list.size(); i++) {
    array[i] = list.get(i);
}

Надеюсь это поможет!


30
ArrayList<String> arrayList = new ArrayList<String>();
Object[] objectList = arrayList.toArray();
String[] stringArray =  Arrays.copyOf(objectList,objectList.length,String[].class);

Используя copyOf, ArrayList для массивов может быть сделано также.


4
Предложите верблюд так, чтобы "objectList = ..." и "stringArray". Кроме того, это Arrays.copyOf ... прописной О.
Джейсон Веден

1
list.toArray () внутренне использует Arrays.copyOf ()
David

25

Начиная с Java-11, можно альтернативно использовать API Collection.toArray(IntFunction<T[]> generator)для достижения того же, что и:

List<String> list = List.of("x","y","z");
String[] arrayBeforeJDK11 = list.toArray(new String[0]);
String[] arrayAfterJDK11 = list.toArray(String[]::new); // similar to Stream.toArray

10

В Java 8:

String[] strings = list.parallelStream().toArray(String[]::new);

12
Это действительно уже содержится в этом ответе . Возможно, добавьте его как редактирование к этому ответу, а не как совершенно новый.
Река

11
Почему parallelStream()вместо просто stream()?
Йори Н.

7

В Java 8 это можно сделать с помощью

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

6

Универсальное решение, позволяющее скрыть любую List<Type>информацию String []:

public static  <T> String[] listToArray(List<T> list) {
    String [] array = new String[list.size()];
    for (int i = 0; i < array.length; i++)
        array[i] = list.get(i).toString();
    return array;
}

Примечание Вы должны override toString()метод.

class Car {
  private String name;
  public Car(String name) {
    this.name = name;
  }
  public String toString() {
    return name;
  }
}
final List<Car> carList = new ArrayList<Car>();
carList.add(new Car("BMW"))
carList.add(new Car("Mercedes"))
carList.add(new Car("Skoda"))
final String[] carArray = listToArray(carList);

5
List <String> list = ...
String[] array = new String[list.size()];
int i=0;
for(String s: list){
  array[i++] = s;
}

13
Это работает, но не очень эффективно, и дублирует функциональность в принятом ответе с дополнительным кодом.
Алан Делимон

5

в случае, если требуется дополнительная манипуляция с данными, для которой пользователь хочет функцию, этот подход не идеален (поскольку требует передачи класса элемента в качестве второго параметра), но работает:

import java.util.ArrayList; import java.lang.reflect.Array;

public class Test {
  public static void main(String[] args) {
    ArrayList<Integer> al = new ArrayList<>();
    al.add(1);
    al.add(2);
    Integer[] arr = convert(al, Integer.class);
    for (int i=0; i<arr.length; i++)
      System.out.println(arr[i]);
  }

  public static <T> T[] convert(ArrayList<T> al, Class clazz) {
    return (T[]) al.toArray((T[])Array.newInstance(clazz, al.size()));
  }
}

5

Если ваше приложение уже использует Apache Commons lib, вы можете немного изменить принятый ответ, чтобы каждый раз не создавать новый пустой массив:

List<String> list = ..;
String[] array = list.toArray(ArrayUtils.EMPTY_STRING_ARRAY);

// or if using static import
String[] array = list.toArray(EMPTY_STRING_ARRAY);

Есть еще несколько предварительно выделенных пустых массивов разных типов в ArrayUtils.

Также мы можем обмануть JVM, чтобы создать для нас пустой массив следующим образом:

String[] array = list.toArray(ArrayUtils.toArray());

// or if using static import
String[] array = list.toArray(toArray());

Но в этом нет никакого преимущества, просто вопрос вкуса, ИМО.


5

Вы можете использовать Iterator<String>для итерации элементов ArrayList<String>:

ArrayList<String> list = new ArrayList<>();
String[] array = new String[list.size()];
int i = 0;
for (Iterator<String> iterator = list.iterator(); iterator.hasNext(); i++) {
    array[i] = iterator.next();
}

Теперь вы можете извлечь элементы из String[]любого цикла.


Я понизил голосование, потому что: 1. нет использования обобщений, которые вынуждают вас использовать 2. .toString()где нет необходимости в явном приведении, 3. вы даже не увеличиваете i, и 4. whileлучше заменить на a for. Предлагаемый код:ArrayList<String> stringList = ... ; String[] stringArray = new String[stringList.size()]; int i = 0; for(Iterator<String> it = stringList.iterator(); it.hasNext(); i++) { stringArray[i] = it.next(); }
Оливье Грегуар

Хорошо, я понял это сейчас. Спасибо.
Ватсал Чавда

Ну, в идеале, вы должны отредактировать свой код, чтобы включить эти комментарии (то есть ... если вы считаете, что они хороши для вашего ответа!). Я не буду рассматривать удаление моего downvote, пока код остается неизменным.
Оливье Грегуар

Я новичок здесь, поэтому я еще не знаю, как здесь все работает. Хотя, ценю вашу помощь, приятель.
Ватсал Чавда

Это намного лучше! Я немного отредактировал, но вы только что узнали, как это работает: предоставьте ответ, улучшите их, получите лавры;)
Оливье Грегуар


4
    List<String> list = new ArrayList<>();
    list.add("a");
    list.add("b");
    list.add("c");
    String [] strArry= list.stream().toArray(size -> new String[size]);

В комментариях я добавил параграф, чтобы объяснить, как работает конвертация. Сначала List преобразуется в поток String. Затем он использует Stream.toArray для преобразования элементов в потоке в массив. В последнем выражении выше «size -> new String [size]» на самом деле является функцией IntFunction, которая выделяет массив String с размером потока String. Утверждение идентично

IntFunction<String []> allocateFunc = size -> { 
return new String[size];
};   
String [] strArry= list.stream().toArray(allocateFunc);

4
Какую ценность добавляет этот ответ? Похоже, что просто повторять другие ответы, не объясняя, почему он отвечает на вопрос.
Ричард

2

Вы можете преобразовать список в массив строк, используя этот метод:

 Object[] stringlist=list.toArray();

Полный пример:

ArrayList<String> list=new ArrayList<>();
    list.add("Abc");
    list.add("xyz");

    Object[] stringlist=list.toArray();

    for(int i = 0; i < stringlist.length ; i++)
    {
          Log.wtf("list data:",(String)stringlist[i]);
    }

1
private String[] prepareDeliveryArray(List<DeliveryServiceModel> deliveryServices) {
    String[] delivery = new String[deliveryServices.size()];
    for (int i = 0; i < deliveryServices.size(); i++) {
        delivery[i] = deliveryServices.get(i).getName();
    }
    return delivery;
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.