ChatGPT解决这个技术问题 Extra ChatGPT

在Java中将数组转换为列表

如何在 Java 中将数组转换为列表?

我使用了 Arrays.asList(),但行为(和签名)不知何故从 Java SE 1.4.2(文档现在存档)更改为 8,我在网络上找到的大多数片段都使用 1.4.2 行为。

例如:

int[] numbers = new int[] { 1, 2, 3 };
Arrays.asList(numbers)

on 1.4.2 返回一个包含元素 1, 2, 3 的列表

在 1.5.0+ 返回包含数组“数字”的列表

在许多情况下,它应该很容易检测到,但有时它可能会被忽视:

Assert.assertTrue(Arrays.asList(numbers).indexOf(4) == -1);
我认为您的示例已损坏:Arrays.asList(new int[] { 1, 2, 3 });肯定没有在 Java 1.4.2 中编译,因为 int[] 不是 Object[]
哦,你可能是对的。在发布之前,我没有 Java 1.4.2 编译器来测试我的示例。现在,在您的评论和乔的回答之后,一切都变得更有意义了。
我认为 Autoboxing 会涵盖从原始到包装 Integer 类的转换。您可以先自己制作演员表,然后上面的 Arrays.asList 代码应该可以工作。
Java 8 的 Stream.boxed() 将负责自动装箱并可用于此目的。请参阅我的答案 below

M
Michael

在您的示例中,这是因为您不能拥有原始类型的 List 。换句话说,List<int> 是不可能的。

但是,您可以使用包装 int 原语的 Integer 类来创建 List<Integer>。使用 Arrays.asList 实用程序方法将您的数组转换为 List

Integer[] numbers = new Integer[] { 1, 2, 3 };
List<Integer> list = Arrays.asList(numbers);

请参阅此code run live at IdeOne.com


或者更简单:Arrays.asList(1, 2, 3);
它怎么知道不创建 List<Integer[]>
@ThomasAhle 它不创建 List 它创建 List 对象。如果您想确保类型安全,请编写: Arrays.asList(spam);
@ThomasAhle,您可以传入 Arrays.asList 0 个或更多元素,以逗号分隔的格式或数组格式,随心所欲。结果将始终相同 - 您指定的元素列表。这是因为方法签名:public static <T>列出<T>作为列表(T...a)。您可以在此处阅读有关可变参数的更多信息docs.oracle.com/javase/tutorial/java/javaOO/…
@ThomasAhle 这是一个好问题。猜测它只是java编译器中的一条规则。例如,以下代码返回一个 ListInteger[] integerArray1 = { 1 }; Integer[] integerArray2 = { 2 }; List<Integer[]> integerArrays = Arrays.asList(integerArray1, integerArray2);
M
Michael

在 Java 8 中,您可以使用流:

int[] numbers = new int[] { 1, 2, 3 };
Arrays.stream(numbers)
      .boxed()
      .collect(Collectors.toList());

这很好,但它不适用于 boolean[]。我想这是因为制作 Boolean[] 一样容易。这可能是一个更好的解决方案。无论如何,这是一个有趣的怪癖。
在 Java 16 中简化为 Arrays.stream(spam).boxed().toList();
T
TylerH

我们不能有 List<int>,因为 int 是原始类型,所以我们只能有 List<Integer>

爪哇 16

Java 16 在 Stream API 上引入了一种名为 toList() 的新方法。这个方便的方法返回一个包含流元素的不可修改的列表。因此,尝试向列表中添加新元素只会导致 UnsupportedOperationException

int[] ints = new int[] {1,2,3,4,5};
Arrays.stream(ints).boxed().toList();

Java 8(整数数组)

int[] ints = new int[] {1,2,3,4,5};
List<Integer> list11 =Arrays.stream(ints).boxed().collect(Collectors.toList()); 

Java 8 及更低版本(整数数组)

Integer[] integers = new Integer[] {1,2,3,4,5};
List<Integer> list21 =  Arrays.asList(integers); // returns a fixed-size list backed by the specified array.
List<Integer> list22 = new ArrayList<>(Arrays.asList(integers)); // good
List<Integer> list23 = Arrays.stream(integers).collect(Collectors.toList()); //Java 8 only

需要 ArrayList 而不是 List?

如果我们想要 List 的特定实现,例如 ArrayList,那么我们可以将 toCollection 用作:

ArrayList<Integer> list24 = Arrays.stream(integers)
                          .collect(Collectors.toCollection(ArrayList::new));

为什么不能对 list21 进行结构修改?

当我们使用 Arrays.asList 时,返回列表的大小是固定的,因为返回的列表不是 java.util.ArrayList,而是在 java.util.Arrays 中定义的私有静态类。因此,如果我们从返回的列表中添加或删除元素,则会抛出 UnsupportedOperationException。因此,当我们要修改列表时,我们应该使用 list22。如果我们有 Java8,那么我们也可以使用 list23

需要明确的是,list21 可以在我们可以调用 list21.set(index,element) 的意义上进行修改,但这个列表可能不会在结构上进行修改,即不能从列表中添加或删除元素。您还可以查看我的这个 answer 以获得更多解释。

如果我们想要一个不可变的列表,那么我们可以将其包装为:

List<Integer> list22 = Collections.unmodifiableList(Arrays.asList(integers));

需要注意的另一点是方法 Collections.unmodifiableList 返回指定列表的不可修改视图。不可修改的视图集合是不可修改的集合,也是支持集合的视图。请注意,对支持集合的更改可能仍然是可能的,并且如果发生更改,它们将通过不可修改的视图可见。

我们可以在 Java 9 和 10 中拥有一个真正不可变的列表。

真正不可变的列表

爪哇 9:

String[] objects = {"Apple", "Ball", "Cat"};
List<String> objectList = List.of(objects);

Java 10(真正的不可变列表):

我们可以使用 Java 9 中引入的 List.of。还有其他方式:

List.copyOf(Arrays.asList(integers)) Arrays.stream(integers).collect(Collectors.toUnmodifiableList());


@BasilBourque 我的意思是不能通过添加/删除来修改列表的结构,但可以更改元素。
确实,我尝试了一些代码并确认调用 List::add & List::remove 失败,Arrays.asList 返回一个列表。该方法的 JavaDoc 在这一点上写得不好并且不清楚。在我的第三次阅读中,我认为“固定大小”一词的意思是说不能添加或删除元素。
R
Roman Nikitchenko

谈到转换方式,这取决于您为什么需要您的 List。如果您只需要它来读取数据。好的,给你:

Integer[] values = { 1, 3, 7 };
List<Integer> list = Arrays.asList(values);

但是,如果你做这样的事情:

list.add(1);

你得到java.lang.UnsupportedOperationException。所以在某些情况下你甚至需要这个:

Integer[] values = { 1, 3, 7 };
List<Integer> list = new ArrayList<Integer>(Arrays.asList(values));

第一种方法实际上并不转换数组,而是像 List 一样“表示”它。但是数组具有所有属性,例如固定数量的元素。请注意,您需要在构造 ArrayList 时指定类型。


“不变性”让我困惑了一会儿。您显然可以更改数组的值。但是,您无法更改其大小。
M
Michael

问题是在 Java 5 中引入了可变参数,不幸的是,Arrays.asList() 也因可变参数版本而过载。因此,Java 5 编译器将 Arrays.asList(numbers) 理解为 int 数组的 vararg 参数。

这个问题在 Effective Java 2nd Ed. 第 7 章第 42 条中有更详细的解释。


我明白发生了什么,但不明白为什么没有记录在案。我正在寻找一种无需重新实现轮子的替代解决方案。
C
Community

我最近不得不将数组转换为列表。后来程序过滤了试图删除数据的列表。当您使用 Arrays.asList(array) 函数时,您创建了一个固定大小的集合:您既不能添加也不能删除。此条目比我能更好地解释问题:Why do I get an UnsupportedOperationException when trying to remove an element from a List?

最后,我不得不进行“手动”转换:

    List<ListItem> items = new ArrayList<ListItem>();
    for (ListItem item: itemsArray) {
        items.add(item);
    }

我想我可以使用 List.addAll(items) 操作添加从数组到列表的转换。


new ArrayList<ListItem>(Arrays.asList(itemsArray)) 会一样
@BradyZhu:当然,上面的答案并不能解决我的固定大小数组的问题,但你基本上是在这里说 RTFM,这总是不好的形式。请说明答案有什么问题或不要发表评论。
检查工具可能会在此处显示警告,并且您已说出原因。无需手动复制元素,请改用 Collections.addAll(items, itemsArray)
刚刚打了这个例外。恐怕Marco13的答案应该是正确的答案。我可能需要通过全年代码来修复所有“asList”异常。
V
Vitaliy Sokolov

更短:

List<Integer> list = Arrays.asList(1, 2, 3, 4);

M
Mincong Huang

使用数组

这是将数组转换为 List 的最简单方法。但是,如果您尝试添加新元素或从列表中删除现有元素,则会抛出 UnsupportedOperationException

Integer[] existingArray = {1, 2, 3};
List<Integer> list1 = Arrays.asList(existingArray);
List<Integer> list2 = Arrays.asList(1, 2, 3);

// WARNING:
list2.add(1);     // Unsupported operation!
list2.remove(1);  // Unsupported operation!

使用 ArrayList 或其他列表实现

您可以使用 for 循环将数组的所有元素添加到 List 实现中,例如 ArrayList

List<Integer> list = new ArrayList<>();
for (int i : new int[]{1, 2, 3}) {
  list.add(i);
}

在 Java 8 中使用 Stream API

您可以将数组转换为流,然后使用不同的收集器收集流:Java 8 中的默认收集器在屏幕后面使用 ArrayList,但您也可以强加您喜欢的实现。

List<Integer> list1, list2, list3;
list1 = Stream.of(1, 2, 3).collect(Collectors.toList());
list2 = Stream.of(1, 2, 3).collect(Collectors.toCollection(ArrayList::new));
list3 = Stream.of(1, 2, 3).collect(Collectors.toCollection(LinkedList::new));

也可以看看:

为什么我们在 Java 中使用自动装箱和拆箱?

何时在 ArrayList 上使用 LinkedList?


A
Alan Bateman

在 Java 9 中,您可以通过新的便利工厂方法 List.of 获得更优雅的使用不可变列表的解决方案:

List<String> immutableList = List.of("one","two","three");

(无耻地从 here 复制)


仅供参考,要了解这背后的一些技术细节,请参阅主要作者 JEP 269: Convenience Factory Methods for Collectionsa talk by Stuart Marks
B
Bhushan

单线:

List<Integer> list = Arrays.asList(new Integer[] {1, 2, 3, 4});

R
Radiodef

如果你的目标是 Java 8(或更高版本),你可以试试这个:

int[] numbers = new int[] {1, 2, 3, 4};
List<Integer> integers = Arrays.stream(numbers)
                        .boxed().collect(Collectors.<Integer>toList());

笔记:

注意 Collectors.<Integer>toList(),这个通用方法可以帮助您避免错误“类型不匹配:无法从 List<Object> 转换为 List<Integer>”。


M
Michael

如果您使用 Apache commons-lang,另一种解决方法:

int[] numbers = new int[] { 1, 2, 3 };
Arrays.asList(ArrayUtils.toObject(numbers));

其中 ArrayUtils.toObjectint[] 转换为 Integer[]


G
Giancarlo Frison

你必须投入到数组中

Arrays.asList((Object[]) array)

java:不兼容的类型:int[] 无法转换为 java.lang.Object[]
@dVaffection 然后转换为 int[]。重要的部分是转换为数组。
D
Danail Tsvetanov

使用番石榴:整数 [] 数组 = { 1, 2, 3};列表<整数> 列表 = Lists.newArrayList(sourceArray);使用 Apache Commons 集合:Integer[] array = { 1, 2, 3}; List list = new ArrayList<>(6); CollectionUtils.addAll(list, array);


T
TylerH

我遇到了同样的问题并编写了一个通用函数,该函数接受一个数组并返回一个具有相同内容的相同类型的 ArrayList:

public static <T> ArrayList<T> ArrayToArrayList(T[] array) {
    ArrayList<T> list = new ArrayList<T>();
    for(T elmt : array) list.add(elmt);
    return list;
}

A
Arpan Saini

给定数组:

    int[] givenArray = {2,2,3,3,4,5};

将整数数组转换为整数列表

一种方法: boxed() -> 返回 IntStream

    List<Integer> givenIntArray1 = Arrays.stream(givenArray)
                                  .boxed()
                                  .collect(Collectors.toList());

第二种方式:将流的每个元素映射到Integer,然后收集

注意:使用 mapToObj,您可以通过将每个 int 元素转换为字符串流、字符流等,方法是将 i 转换为 (char)i

    List<Integer> givenIntArray2 = Arrays.stream(givenArray)
                                         .mapToObj(i->i)
                                         .collect(Collectors.toList());

将一种数组类型转换为另一种类型示例:

List<Character> givenIntArray2 = Arrays.stream(givenArray)
                                             .mapToObj(i->(char)i)
                                             .collect(Collectors.toList());

H
Hitesh Garg

所以这取决于你正在尝试哪个 Java 版本-

爪哇 7

 Arrays.asList(1, 2, 3);

或者

       final String arr[] = new String[] { "G", "E", "E", "K" };
       final List<String> initialList = new ArrayList<String>() {{
           add("C");
           add("O");
           add("D");
           add("I");
           add("N");
       }};

       // Elements of the array are appended at the end
       Collections.addAll(initialList, arr);

或者

Integer[] arr = new Integer[] { 1, 2, 3 };
Arrays.asList(arr);

在 Java 8 中

int[] num = new int[] {1, 2, 3};
List<Integer> list = Arrays.stream(num)
                        .boxed().collect(Collectors.<Integer>toList())

参考 - http://www.codingeek.com/java/how-to-convert-array-to-list-in-java/


D
DevilCode

请您改进这个答案,因为这是我使用的,但我不是 100% 清楚。它工作正常,但 intelliJ 添加了新的 WeatherStation[0]。为什么是 0 ?

public WeatherStation[] removeElementAtIndex(WeatherStation[] array, int index) { List list = new ArrayList(Arrays.asList(array)); list.remove(索引); return list.toArray(new WeatherStation[0]); }


我们为什么要改进您的问题?要么你知道答案,要么你不知道。你当然应该提供一个真正帮助他人的答案,而不是一个比帮助更多的混淆的答案。
i
ivenpoker

从 Java 8 开始,应该执行以下操作

int[] temp = {1, 2, 3, 4, 5};
List<Integer> tempList = Arrays.stream(temp).boxed().collect(Collectors.toList());

D
Dishant Tanwar

使用它来将 Array arr 转换为 List。 Arrays.stream(arr).collect(Collectors.toList());

定义将数组转换为列表的泛型方法的示例:

public <T> List<T> fromArrayToList(T[] a) {   
    return Arrays.stream(a).collect(Collectors.toList());
}

y
yousef

使用两行代码将数组转换为列表,如果在整数值中使用它,则必须对原始数据类型使用自动装箱类型

  Integer [] arr={1,2};
  List<Integer> listInt=Arrays.asList(arr);

这是最简单的方法,应标记为正确答案。谢谢@yousef
A
ATOMP

如果您正在尝试针对内存等进行优化(并且不想引入外部库),那么实现自己的不可变“数组视图列表”比您想象的要简单 – 您只需要扩展 { 1}。

class IntArrayViewList extends AbstractList<Integer> {
    int[] backingArray;
    int size;

    IntArrayViewList(int[] backingArray, int size) {
        this.backingArray = backingArray;
        this.size = size;
    }

    public Iterator<Integer> iterator() {
        return new Iterator<Integer>() {
            int i = 0;

            @Override
            public boolean hasNext() {
                return i < size;
            }

            @Override
            public Integer next() {
                return get(i++);
            }
        };
    }

    public int size() {
        return size;
    }

    public Integer get(int i) {
        return backingArray[i];
    }
}