Java: Как я могу динамически создать массив указанного типа в зависимости от типа объекта?

Я хотел бы взять переданный список, который, как я знаю, является однородным и из него создает массив того же типа, что и элементы внутри него.

Что-то вроде...

List<Object> lst = new ArrayList<Object>;

lst.add(new Integer(3));

/// somewhere else ...

assert(my_array instanceof Integer[]);

Ответ 1

Преобразование будет выполняться во время выполнения, в то время как тип теряется во время компиляции. Поэтому вы должны сделать что-то вроде:

public <T> T[] toArray(List<T> list) {
    Class clazz = list.get(0).getClass(); // check for size and null before
    T[] array = (T[]) java.lang.reflect.Array.newInstance(clazz, list.size());
    return list.toArray(array);
}

Но будьте осторожны, что 3-я строка выше может выдать исключение - это не типы.

Ответ 2

Этот метод безопасен по типу и обрабатывает некоторые нули (по крайней мере один элемент должен быть не нулевым).

public static Object[] toArray(Collection<?> c)
{
  Iterator<?> i = c.iterator();
  for (int idx = 0; i.hasNext(); ++idx) {
    Object o = i.next();
    if (o != null) {
      /* Create an array of the type of the first non-null element. */
      Class<?> type = o.getClass();
      Object[] arr = (Object[]) Array.newInstance(type, c.size());
      arr[idx++] = o;
      while (i.hasNext()) {
        /* Make sure collection is really homogenous with cast() */
        arr[idx++] = type.cast(i.next());
      }
      return arr;
    }
  }
  /* Collection is empty or holds only nulls. */
  throw new IllegalArgumentException("Unspecified type.");
}

Ответ 3

java.lang.reflect.Array.newInstance(Class<?> componentType, int length)

Ответ 4

Если вам необходимо динамически создавать массив на основе типа, известного только во время выполнения (скажем, вы делаете отражение или генерики), вы, вероятно, захотите Array.newInstance