Collections的copy()方法和ArrayList的大小问题

本文对比了Collections的copy方法与ArrayList构造函数的拷贝行为,解释了深拷贝与浅拷贝的区别及其应用场景。并详细介绍了如何正确设置目标列表容量避免异常。

 Collections的copy()方法和ArrayList的大小问题

偶然看到了Collections的copy(List desc,List src)方法.当时就想这个方法和初始化一个List desc=new ArrayList(List c)【参数必须实现Collection接口】的区别。

两者的差别很大,后者是一个浅拷贝,只是对源list的元素进行拷贝,拷贝的只是引用。拷贝后两个list的元素(引用)不同,但是引用所指向的对象是一样的。即是两个list的每个元素指向的还是通一内存。然而前者是深拷贝,不光拷贝的是src的元素(引用),src内每个元素的所指向的对象都进行一次拷贝。即是两个list的每个元素所指向的不是同一内存。

使用后者进行拷贝的结果是:当你的desc链表发生改变时,src也将会随之改变。
使用前者进行拷贝时你又必须要注意目标链表的长度必须要比源链表的长度大或者相等。

举例如下:
List src1=new ArrayList(3)
src1.add("a");
src2.add("b");
src3.add("c");

如果你使用下面方法copy链表
/*******************************/
List des1=new ArrayList(3);
Collections.copy(des1,src1);
/*******************************/
将会出错,抛出数组越界异常。
当时我怎么想都想不明白为什么,明明已经设置了长度为3,为什么还会出错!
后来打印出des1.size()才知道des1的长度为0;3表示的是这个List的容纳能力为3,并不是说des1中就有了3个元素。查看api才知道,它的capacity(容纳能力大小)可以指定(最好指定)。而初始化时size的大小永远默认为0,只有在进行add和remove等相关操作时,size的大小才变化。然而进行copy()时候,首先做的是将desc1的size和src1的size大小进行比较,只有当desc1的size大于或者等于src1的size时才进行拷贝,否则抛出IndexOutOfBoundsException异常。

所以可以通过下面的方法指定目标desc的大小
/*******************************/
List des1=new ArrayList(Array.asList(new object[src1.size]));//注意:new ArrayList(Collection col)参数必须要实现Collection 接口。
Collections.copy(des1,src1);
/*******************************/
执行第一句后size的大小是3,其实它是对一个空数组的浅拷贝。

### Java `Collections.copy` 方法使用示例 #### 使用示例 为了展示如何正确使用 `Collections.copy` 方法,考虑如下代码片段: ```java import java.util.*; public class CopyExample { public static void main(String[] args) { List<String> source = Arrays.asList("apple", "banana", "orange"); // 创建目标列表并填充适当数量的占位符元素 List<String> destination = new ArrayList<>(Arrays.asList(null, null, null)); // 执行复制操作 Collections.copy(destination, source); // 输出结果 System.out.println("Destination after copy: " + destination); } } ``` 这段代码展示了如何从源列表向目的地列表复制元素。注意,在调用 `Collections.copy` 之前,目的列表必须已经被初始化,并且其大小应至少等于源列表中的元素数目[^2]。 #### 常见问题及解决方案 当使用 `Collections.copy` 时可能会遇到一些典型错误,以下是几个需要注意的地方及其对应的解决办法: - **ClassCastException**: 如果尝试将不兼容类型的对象放入经过类型检查的集合中,则会抛出此异常。例如,试图往整数类型的已验证列表里加入字符串会导致运行期转换失败[^3]。 - **IndexOutOfBoundsException 或 ArrayIndexOutOfBoundsException**: 当目的列表长度小于源列表时发生。为了避免这种情况,确保在执行拷贝前调整好目的列表的容量,使其能够容纳所有来自源的数据项[^4]。 - **NullPointerException**: 若任一参数为null也会引发该异常。因此,在传递给 `Collections.copy` 的两个列表都不能为空。 通过遵循上述指导原则可以有效减少因误用 API 而产生的编程错误。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值