ArrayList和Arrays.ArrayList

本文详细解析了Java中ArrayList与Arrays.ArrayList的区别,展示了如何通过不同方法转换数组到列表,以及这些操作对原始数组的影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

java中的ArrayList内部是由数组实现的,但是正因为这一点,ArrayList与数组打交道的时候应该要注意一些问题。
测试代码1
public static void main(String[] args) {
        //创建一个字符串数组
        String[] words = "And that is how we know the Earth to be banana-shaped".split(" ");
        //将该数组通过Arrays.asList方法,通过ArrayList的构造器传递给该list
        List<String> list = Arrays.asList(words);
        
        System.out.println("list改变前数组的第一个元素 : " + words[0]);
        System.out.println("list改变前的list" + list);
        
        /*
        *  改变 list 会影响原数组
        * */
        //将list中的元素换位
        String temp = list.get(0);
        list.set(0,list.get(1));
        list.set(1,temp);
        System.out.println("list改变后的list" + list);
        System.out.println("list改变后数组的第一个元素 : " + words[0]);
    }
输出结果:
list改变前数组的第一个元素 : And
list改变前的list[And, that, is, how, we, know, the, Earth, to, be, banana-shaped]
list改变后的list[that, And, is, how, we, know, the, Earth, to, be, banana-shaped]
list改变后数组的第一个元素 : that

可见,修改list之后,原数组也被跟着改变了。

原因:

asList()方法生成一个ArrayList,但是需要注意的是,这个ArrayList并不是ArrayList类,而是Arrays类里面的嵌套类Arrays.ArrayList类,该嵌套类的信息类标签和构造器如下:

private static class ArrayList<E> extends AbstractList<E> implements RandomAccess, java.io.Serializable

private final E[] a;

ArrayList(E[] array) {
    a = Objects.requireNonNull(array);
}

可见Arrays.ArrayList有一个数组成员 E[] a。

只有这个Arrays.ArrayList的构造器能够接受数组作为构造器参数,ArrayList的构造器是只能接受Collection的。
Arrays.ArrayList的构造器直接让它的数组成员a指向了传进来的array。

也就是说,我们后面玩的ArrayList其实是Arrays.ArrayList,他是没有add()方法的,并且修改元素也是通过修改之前传递进去的固定长度数组来实现,这就是为什么修改它的元素会直接影响传进来的数组。

这个嵌套类虽然也是继承自AbstractList,但是它没实现add()方法,用的还是AbstractList方法,因此一旦调用add()方法会报错UnSupportedOperationException。

测试代码2

ArrayList中也有一个成员数组,对ArrayList的操作是通过对该数组的操作来实现的

transient Object[] elementData; 

但是如果将Arrays.asList(String…)方法用Collections.addAll(Collection,String…)代替,
对list的操作不会影响原数组,因为这个方法是把原数组的数据拷贝过来,而不是直接拷贝引用。

public static void main(String[] args) {
            //创建一个字符串数组
            String[] words = "And that is how we know the Earth to be banana-shaped".split(" ");
            //将该数组通过Arrays.asList方法,通过ArrayList的构造器传递给该list
            List<String> list = new ArrayList<>();
            Collections.addAll(list,words);
            
            System.out.println("list改变前数组的第一个元素 : " + words[0]);
            System.out.println("list改变前的list" + list);
            
            /*
            *  改变 list 不会影响原数组
            * */
            //将list中的元素换位
            String temp = list.get(0);
            list.set(0,list.get(1));
            list.set(1,temp);
            System.out.println("list改变后的list" + list);
            System.out.println("list改变后数组的第一个元素 : " + words[0]);
        }

输出结果:

list改变前数组的第一个元素 : And
list改变前的list[And, that, is, how, we, know, the, Earth, to, be, banana-shaped]
list改变后的list[that, And, is, how, we, know, the, Earth, to, be, banana-shaped]
list改变后数组的第一个元素 : And
总结

ArrayList是一个单独的类,Arrays.ArrayList是Arrays的一个嵌套类。前者的对象是我们常见的长度可变的容器,而后者是Arrays类利用asList()方法产生的一个长度不可变的容器 ( 因为没实现add()等方法 ),两者的实现是不同的。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值