当前位置: 首页 > 知识库问答 >
问题:

java list.toArray 方法?

谢志用
2023-10-16
        List<Integer> list = List.of(12, 34, 56);        Integer[] array = list.toArray(new Integer[]{1, 2, 1, 43, 32, 1});        System.out.println(Arrays.toString(array)); 

输出: [12, 34, 56, null, 32, 1]


不应该是: [12, 34, 56, null, null, null] 吗?
这是为什么?
版本:GraalVM for JDK 17

共有2个答案

蒋寒
2023-10-16

这个问题看java的源代码就好,主要不理解的应该就是
Integer[] array = list.toArray(new Integer[]{1, 2, 1, 43, 32, 1});
这行代码。我本地是jdk8的环境,但是运行结果是一样,进入ArrayList源码中

 public <T> T[] toArray(T[] a) {        if (a.length < size)            // Make a new array of a's runtime type, but my contents:            return (T[]) Arrays.copyOf(elementData, size, a.getClass());        System.arraycopy(elementData, 0, a, 0, size);        if (a.length > size)            a[size] = null;        return a;    }

可以看到,a的长度如果比list长度小,就返回list的数组形式。
如果a的长度比list长度长,就把list的元素拷贝到a的指定位置,因为a的超长部分已经有值了,所以是二者的组合
最后可能为了分割吧,把list结尾换成了个null。

写代码组合的时候不建议用这么不好理解的Api。而且碰到问题看源码即可

詹夕
2023-10-16

看一下toArray方法的源码文档:

* If the list fits in the specified array with room to spare (i.e.,* the array has more elements than the list), the element in the array* immediately following the end of the list is set to <tt>null</tt>.* (This is useful in determining the length of the list <i>only</i> if* the caller knows that the list does not contain any null elements.)

意思就是如果数组的元素比列表多,数组中紧跟在列表末尾后面那一位置的元素设置为null,
如果调用者知道列表不包含任何null元素的情况下,方便推断得出列表的真实长度
以ArrayList为例,它的实现如下:

@SuppressWarnings("unchecked")public <T> T[] toArray(T[] a) {    if (a.length < size)        // Make a new array of a's runtime type, but my contents:        return (T[]) Arrays.copyOf(elementData, size, a.getClass());    System.arraycopy(elementData, 0, a, 0, size);    if (a.length > size)//如果数组的长度大于列表,把列表后面的第一个位置置为null        a[size] = null;    return a;}

你题目里这个of生成的应该是一个UnmodifiableList? 它的实现也差不多,总之也是只把紧跟着列表后面的那个元素置为null:

@SuppressWarnings("unchecked")public <T> T[] toArray(T[] a) {    // We don't pass a to c.toArray, to avoid window of    // vulnerability wherein an unscrupulous multithreaded client    // could get his hands on raw (unwrapped) Entries from c.    Object[] arr = c.toArray(a.length==0 ? a : Arrays.copyOf(a, 0));    for (int i=0; i<arr.length; i++)        arr[i] = new UnmodifiableEntry<>((Map.Entry<? extends K, ? extends V>)arr[i]);    if (arr.length > a.length)        return (T[])arr;    System.arraycopy(arr, 0, a, 0, arr.length);    if (a.length > arr.length)        a[arr.length] = null;//看这里看这里    return a;}
 类似资料:
  • 例如,我如何做到这一点? 和 像这样的东西,我试着在网上搜索,但我找不到任何关于这方面的教程,所以我在这里问。

  • 只是想知道为什么 调用world\u开关: 结果没有任何变化,我想增加或减少实例变量@pos\u X或@pos\u Y 这是我的初始化方法 这就是我如何创建类的实例 我们将不胜感激

  • 本文向大家介绍iOS hitTest方法&pointInside方法相关面试题,主要包含被问及iOS hitTest方法&pointInside方法时的应答技巧和注意事项,需要的朋友参考一下 hitTest方法 当事件传递给控件的时候,就会调用控件的这个方法,去寻找最合适的view point:当前的触摸点,point这个点的坐标系就是方法调用者   pointInside方法 作用:判断当前这个

  • 书籍与公开课 我喜欢看一些有关基础原理的书,如操作系统、计算机网络等。但是最近在看《计算机网络:自顶向下方法》的时候发现,这些书都太大而全了,让人看着很吃力,而且抓不住重点。 知名大学都有公开课,视频比枯燥的书要更好,也讲的更有侧重点。我觉得以后可以跟着公开课来看书: 如果不知道某个科目要看什么书,可以跟着公开课学习,看参考教材 如果已经确定要读某本经典书,可以先找有没有公开课 这些公开课可以在知

  • 主要内容:Python类实例方法,Python类方法,Python类静态方法和类属性一样,类方法也可以进行更细致的划分,具体可分为 类方法、 实例方法和 静态方法。 和类属性的分类不同,对于初学者来说,区分这 3 种类方法是非常简单的,即采用 @classmethod 修饰的方法为类方法;采用 @staticmethod 修饰的方法为静态方法;不用任何修改的方法为实例方法。 其中 @classmethod 和 @staticmethod 都是函数装饰器,后续章节会对其做详

  • Methods 方法 Pointers vs. Values 指针 vs. 值 As we saw with ByteSize, methods can be defined for any named type (except a pointer or an interface); the receiver does not have to be a struct. 正如 ByteSize 那样

  • 方法 是关联了特定类型的函数。类,结构体以及枚举都能定义实例方法,方法封装了给定类型特定的任务和功能。类,结构体和枚举同样可以定义类型方法,这是与类型本身关联的方法。类型方法与 Objective-C 中的类方法相似。 事实上在 结构体和枚举中定义方法是 Swift 语言与 C 语言和 Objective-C 的主要区别。在 Objective-C 中,类是唯一能定义方法的类型。但是在 Swift

  • 本页包含内容: 实例方法(Instance Methods) 类型方法(Type Methods) 方法是与某些特定类型相关联的函数。类、结构体、枚举都可以定义实例方法;实例方法为给定类型的实例封装了具体的任务与功能。类、结构体、枚举也可以定义类型方法;类型方法与类型本身相关联。类型方法与 Objective-C 中的类方法(class methods)相似。 结构体和枚举能够定义方法是 Swif