Java List 数据拷贝

今天想到了List数据的拷贝的问题,写了一个程序来测试,发现不同的写法会产生浅拷贝和深拷贝,如果写法不对,指向同一个引用,会存在问题的。下面的不同写法展示了不同的结果,看完示例代码后应该会明白不同写法的作用。

        import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

class Person implements Cloneable,Serializable{
	private String name;

	public Person(String name) {
		this.name = name;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}
	@Override
	public String toString() {
		return this.name;
	}
	@Override
	protected Person clone() throws CloneNotSupportedException {
		return (Person)super.clone();
	}

}

/**
 * 
 * 测试不同写法完成list数据拷贝的效果
 *
 */
public class JavaListCopy {
	public static void main(String args[]) throws IOException, CloneNotSupportedException, ClassNotFoundException {
        List<Person> list1 = new ArrayList<Person>();
        
        list1.add(new Person("aaa"));
        list1.add(new Person("bbb"));
        // 第一种方法:通过构造函数来实现,它里面调用了Arrays.copyOf(),copyOf又调用了System.arraycopy
        List<Person> list2 = new ArrayList<Person>(list1);
        list2.get(0).setName("ccc");
        
        System.out.println("list1 ==>"+list1);
        System.out.println("list2 ==>"+list2);
        /** 此时list1 和list2 输出相同,说明是引用*/
        // 第二种方法:addAll方法,它调用了System.arraycopy
        // System.arraycopy:源数据-->目的数据
        // Arrays.copyOf():源数据及长度,数据类型,它里面调用了System.arraycopy
        List<Person> list3 = new ArrayList<Person>();
        list3.addAll(list1);
        list3.get(0).setName("ddd");
        
        System.out.println("list1 ==>"+list1);
        System.out.println("list3 ==>"+list3);
        /** 此时list1 和list3 输出相同,说明是引用*/
        
        // 第三种方法:实现clone来实现
        List<Person> list4 = new ArrayList<Person>();
        for(int i = 0; i< list1.size();i++){
        	list4.add(list1.get(i).clone());
        }
        list4.get(0).setName("eee");
        System.out.println("list1 ==>"+list1);
        System.out.println("list4 ==>"+list4);
        /** 此处clone了,但要注意实现了cloneable接口只能对原始类型和String进行clone,如果是引用类型,则要进行深clone,复制内容或者对象再次clone*/
        
        // 第四种方法:通过序列化来实现
        List<Person> list5 = null;
        OutputStream bo = new ByteArrayOutputStream();
        ObjectOutputStream oo = new ObjectOutputStream(bo);
	oo.writeObject(list1);
		
	InputStream bi = new ByteArrayInputStream(((ByteArrayOutputStream) bo).toByteArray()); 
	ObjectInputStream oi = new ObjectInputStream(bi);
        list5 = (List<Person>) oi.readObject();
        list5.get(0).setName("ggg");
        System.out.println("list1 ==>"+list1);
        System.out.println("list5 ==>"+list5);       
        /** 通过这种方法也能实现深拷贝*/

        // 第五种方法: Collection.copy,它的实现过程是通过遍历然后set进入
        List<Person> list6 = new ArrayList<Person>(Arrays.asList(new Person[list1.size()]));
        
        Collections.copy(list6, list1);
        list6.get(0).setName("hhh");
        System.out.println("list1 ==>"+list1);
        System.out.println("list6 ==>"+list6);  
        /** list1 和list6 输出结果是相同的*/
	}
}


      
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值