ArrayList类中的toArray()与Arrays类中的toArray()方法注意问题

1.Arrays类是一个 final 类,其类中含有一个私有的 ArrayList, ArrayList的定义与 java.util.ArrayList中的定义相同。


java源码:

[java]  view plain  copy
  1. public class ArrayList<E> extends AbstractList<E>  
  2.         implements List<E>, RandomAccess, Cloneable, java.io.Serializable  
  3. {  


2. Arrays调用 toArray()时, 调用的是 ArrayList 中数据数组的 clone, 当调用 toArray(T[])时,则使用 System.arraycopy()。


Arrays.java

[java]  view plain  copy
  1. <span style="font-size:18px;"> <span style="white-space:pre">   </span>@Override  
  2.         public Object[] toArray() {  
  3.             return a.clone();  
  4.         }  
  5.   
  6.         @Override  
  7.         @SuppressWarnings("unchecked")  
  8.         public <T> T[] toArray(T[] a) {  
  9.             int size = size();  
  10.             if (a.length < size)  
  11.                 return Arrays.copyOf(this.a, size,  
  12.                                      (Class<? extends T[]>) a.getClass());  
  13.             System.arraycopy(this.a, 0, a, 0, size);  
  14.             if (a.length > size)  
  15.                 a[size] = null;  
  16.             return a;  
  17.         }</span>  

3. ArrayList 则是调用 Arrays.copyof, 再调用Arrays类中的 System.arraycopp。

ArrayList.java

[java]  view plain  copy
  1. <pre name="code" class="java">public Object[] toArray() {  
  2.         return Arrays.copyOf(elementData, size);  


 
 
 
 

Arrays.java
[java]  view plain  copy
  1. <span style="font-size:18px;"public static <T> T[] copyOf(T[] original, int newLength) {  
  2.         return (T[]) copyOf(original, newLength, original.getClass());  
  3.     }</span>  


4. 分析一下运行结果。

[java]  view plain  copy
  1. <span style="font-size:18px;"><span style="white-space:pre">    </span>@Test  
  2.     public void test() {  
  3.         // 使用 Arrays类中 toArray  
  4.         StringBuilder sb = new StringBuilder("a");  
  5.         Object[] o = new Object[] {sb, "b""c"};  
  6.         Arrays.asList(o).toArray();  
  7.         System.out.println(sb.toString());  
  8.         sb.setCharAt(0'X');  
  9.         System.out.println(sb.toString());  
  10.           
  11.         // 使用 ArrayList 类中的 toArray  
  12.         StringBuilder sb2 = new StringBuilder("a");  
  13.         Object[] o3 = new Object[] {sb2, "b""c"};  
  14.         new ArrayList<>(Arrays.asList(o3)).toArray();  
  15.         System.out.println(sb2.toString());  
  16.         sb.setCharAt(0'X');  
  17.         System.out.println(sb2.toString());  
  18.     }</span>  

5.结果:a X a a


6. 使用 T 方法时,两个结果都是一样的,因为都是使用 System.arraycopy 方法,而不是使用 clone() 方法。


ArrayList.java

[java]  view plain  copy
  1. <span style="font-size:18px;">public <T> T[] toArray(T[] a) {  
  2.         if (a.length < size)  
  3.             // Make a new array of a's runtime type, but my contents:  
  4.             return (T[]) Arrays.copyOf(elementData, size, a.getClass());  
  5.         System.arraycopy(elementData, 0, a, 0, size);  
  6.         if (a.length > size)  
  7.             a[size] = null;  
  8.         return a;  
  9.     }  
  10. </span>  


Arrays.java

[java]  view plain  copy
  1. <span style="font-size:18px;"public <T> T[] toArray(T[] a) {  
  2.             int size = size();  
  3.             if (a.length < size)  
  4.                 return Arrays.copyOf(this.a, size,  
  5.                                      (Class<? extends T[]>) a.getClass());  
  6.             System.arraycopy(this.a, 0, a, 0, size);  
  7.             if (a.length > size)  
  8.                 a[size] = null;  
  9.             return a;  
  10.         }</span>  

7. 总结
推荐使用带有参数的 toArray(T[] a)  方法,这样就避免了浅克隆带来的问题。
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ArrayList的toArray方法用于将列表中的元素转换成一个数组。它有两个重载方法: ```java public Object[] toArray(); public <T> T[] toArray(T[] a); ``` 第一个方法返回一个Object类型的数组,其中包含了列表中所有元素。第二个方法返回一个指定类型的数组,其中包含了列表中所有元素。 下面是ArrayList的toArray方法的源码: ```java public Object[] toArray() { return Arrays.copyOf(elementData, size); } @SuppressWarnings("unchecked") public <T> T[] toArray(T[] a) { if (a.length < size) return (T[]) Arrays.copyOf(elementData, size, a.getClass()); System.arraycopy(elementData, 0, a, 0, size); if (a.length > size) a[size] = null; return a; } ``` 首先看第一个方法,它的实现很简单,就是使用Arrays.copyOf方法将elementData数组中的元素复制到一个新的数组中,并返回这个新的数组。这个新数组的长度就是列表的长度,也就是size属性的值。 第二个方法稍微复杂一些。它首先判断输入的数组a是否足够长,能够容纳列表中的所有元素。如果不够长,就创建一个新的数组,长度为size,类型和a相同。然后将elementData数组中的元素复制到这个新的数组中,并返回这个新的数组。 如果a足够长,就直接将elementData数组中的元素复制到a数组中,并返回a数组。如果a的长度比size大,那么在a的最后一个元素位置上设置为null。这是因为如果a数组的长度比列表的长度还大,那么a数组中多出来的部分应该被设置为null。 总之,ArrayList的toArray方法就是将列表中的元素复制到一个新的数组中,并返回这个新的数组。如果输入的数组足够长,就直接将元素复制到输入的数组中。否则就创建一个新的数组。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值