java深浅复制以及引用传递

之前探讨过Java数组的深复制问题,现在来说说<一些不靠谱的java.util.List深复制方法>。为什么不说<靠谱的深复制方法>呢?因为在寻找探索<靠谱的深复制方法>的过程中,我发现了这些不靠谱的方法,写下来是希望给自己和他人提个醒,不要犯这样的错误。

这是下面要频繁使用的一个JavaBean

class Person implements Serializable{  
    private int age;  
    private String name;  
      
    public Person(){};  
    public Person(int age,String name){  
        this.age=age;  
        this.name=name;  
    }  
      
    public int getAge() {  
        return age;  
    }  
    public void setAge(int age) {  
        this.age = age;  
    }  
    public String getName() {  
        return name;  
    }  
    public void setName(String name) {  
        this.name = name;  
    }  
      
    public String toString(){  
        return this.name+"-->"+this.age;  
    }  
      
}

后台打印List集合的一个静态方法

public static <T> void printList(List<T> list){  
    System.out.println("---begin---");  
    for(T t : list){  
        System.out.println(t);  
    }  
    System.out.println("---end---");  
}

后台打印数组的一个静态方法

public static <T> void printArray(T[] array){  
    System.out.println("---begin---");  
    for(T t : array){  
        System.out.println(t);  
    }  
    System.out.println("---end---");  
}

这是数据源集合,下面将通过各种方法企图来深复制该List集合中的元素

List<Person> srcList=new ArrayList<Person>();  
Person p1=new Person(20,"123");  
Person p2=new Person(21,"ABC");  
Person p3=new Person(22,"abc");  
srcList.add(p1);  
srcList.add(p2);  
srcList.add(p3);

1、遍历循环复制

List<Person> destList=new ArrayList<Person>(srcList.size());  
for(Person p : srcList){  
    destList.add(p);  
}  
          
printList(destList);  
srcList.get(0).setAge(100);  
printList(destList);

上面的代码在add时候,并没有new Person()操作。因此,在srcList.get(0).setAge(100);破坏源数据时,目标集合destList中元素的输出同样受到了影响,原因是浅复制造成的。

后台输出结果代码 收藏代码 ---begin---
123-->20
ABC-->21
abc-->22
---end---
---begin---
123-->100
ABC-->21
abc-->22
---end---

2、使用List实现类的构造方法

Java代码 收藏代码

List<Person> destList=new ArrayList<Person>(srcList);  
printList(destList);  
srcList.get(0).setAge(100);  
printList(destList);

通过ArrayList的构造方法来复制集合内容,同样是浅复制,在修改了源数据集合后,目标数据集合对应内容也发生了改变。在查阅资料的过程中,看到有人说这种方式 能实现深复制,其实这是不对的。对于某些特殊的元素,程序运行的结果形似深复制,其实还是浅复制。具体一会儿再说。

后台打印输出代码 收藏代码 ---begin---
123-->20
ABC-->21
abc-->22
---end---
---begin---
123-->100
ABC-->21
abc-->22
---end---

3、使用list.addAll()方法

Java代码 收藏代码

List<Person> destList=new ArrayList<Person>();  
destList.addAll(srcList);  
printList(destList);  
srcList.get(0).setAge(100);  
printList(destList);

java.util.list.addAll()方法同样是浅复制

后台打印输出代码 收藏代码 ---begin---
123-->20
ABC-->21
abc-->22
---end---
---begin---
123-->100
ABC-->21
abc-->22
---end---

4、使用System.arraycopy()方法

Java代码 收藏代码

Person[] srcPersons=srcList.toArray(new Person[0]);  
Person[] destPersons=new Person[srcPersons.length];  
System.arraycopy(srcPersons, 0, destPersons, 0, srcPersons.length);  
//destPersons=srcPersons.clone();  
  
printArray(destPersons);  
srcPersons[0].setAge(100);  
printArray(destPersons);  
  
List<Person> destList=Arrays.asList(destPersons);  
printList(destList);

这种方式虽然比较变态,但是起码证明了System.arraycopy()方法和clone()是不能对List集合进行深复制的。

5、使用序列化方法(相对靠谱的方法)

public static <T> List<T> deepCopy(List<T> src) throws IOException, ClassNotFoundException {  
    ByteArrayOutputStream byteOut = new ByteArrayOutputStream();  
    ObjectOutputStream out = new ObjectOutputStream(byteOut);  
    out.writeObject(src);  
  
    ByteArrayInputStream byteIn = new ByteArrayInputStream(byteOut.toByteArray());  
    ObjectInputStream in = new ObjectInputStream(byteIn);  
    @SuppressWarnings("unchecked")  
    List<T> dest = (List<T>) in.readObject();  
    return dest;  
}

Java代码 收藏代码

List<Person> destList=deepCopy(srcList);  
printList(destList);  
srcList.get(0).setAge(100);  
printList(destList);

这是比较靠谱的做法,听说是国外某位程序大师提出来的。实际运行的结果也同样是正确的。 后台打印输出代码 收藏代码 ---begin---
123-->20
ABC-->21
abc-->22
---end---
---begin---
123-->20
ABC-->21
abc-->22
---end---

其实,上面这些不靠谱List深复制的做法在某些情况是可行的,这也是为什么有些人说这其中的一些做法是可以实现深复制的原因。哪些情况下是可行(本质上可能还是不靠谱)的呢?比如List<String>这样的情况。我上面使用的是List<Person>,它和List<String>的区别就在于Person类和String类的区别,Person类提供了破坏数据的2个setter方法。因此,在浅复制的情况下,源数据被修改破坏之后,使用相同引用指向该数据的目标集合中的对应元素也就发生了相同的变化。 因此,在需求要求必须深复制的情况下,要是使用上面提到的方法,请确保List<T>中的T类对象是不易被外部修改和破坏的。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Python中,深复制和浅复制是两种不同的复制方式。 浅复制是创建一个新对象,这个对象有着原始对象的引用,但是它并不复制原始对象的数据。在Python中,使用copy()方法可以实现浅复制。 深复制是创建一个新对象,这个对象有着原始对象数据的完整复制。在Python中,使用deepcopy()方法可以实现深复制。 需要注意的是,对于不可变对象(如字符串、数字、元组等),无论是深复制还是浅复制都是一样的结果。而对于可变对象(如列表、字典等),深复制复制整个对象,包括对象内部的所有数据,而浅复制只会复制对象的引用,而不会复制对象内部的数据。 下面是一个例子,展示了浅复制和深复制的区别: ``` import copy # 浅复制 a = [1, 2, [3, 4]] b = copy.copy(a) a[2][0] = 5 print(a) # [1, 2, [5, 4]] print(b) # [1, 2, [5, 4]] # 深复制 c = copy.deepcopy(a) a[2][1] = 6 print(a) # [1, 2, [5, 6]] print(c) # [1, 2, [5, 4]] ``` 在上面的例子中,我们首先使用浅复制创建了一个列表b,然后修改了a列表中的一个元素。可以看到,由于浅复制复制了对象的引用,所以修改a列表中的元素同时也修改了b列表中的元素。 然后我们使用深复制创建了一个列表c,再次修改了a列表中的一个元素。可以看到,由于深复制复制的是整个对象,所以修改a列表中的元素并没有影响到c列表中的元素。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值