当前位置:首页 » 编程语言 » 深度克隆java

深度克隆java

发布时间: 2022-02-27 15:37:19

java中深克隆与浅克隆的区别

深克隆与浅克隆

大家知道,对象是互相引用的,即对象中可能包含了另一个对象的引用,举例如:有一个Order对象,Order对象中又包含了LineItems对象,然后LineItems对象又包含了Item对象。

好了,现在我有一个Order对象order1,它包含了一个LineItems对象items,这表示的是有一个订单order1,订单的内容是items。

好的,现在有另一个客户想要一份订单,内容跟order1完全一样,那么在系统的逻辑层我们怎么做呢?很简单,order2=order1.clone(). 我们知道clone方法是在内存中生成一个新的对象,而不是只得到原对象的引用。这时候,有人说话了:“哦,明白了我们对order2的成员变量进行修改,是不会影响order1的。” 很可惜,这句话只对了一半。

假设order类有一个成员变量name,当然改变order2.name不会影响order1.name,因为他们在不同的内存区域。但是如果改变 order1.items呢?很遗憾,简单地使用order1.clone,是会影响到order2.items的。原因很简单,就是因为clone方法默认的是浅克隆,即不会克隆对象引用的对象,而只是简单地复制这个引用。所以在上例中,items对象在内存中只有一个,order1和order2都指向它,任何一个对象对它的修改都会影响另一个对象。

那相对浅克隆,深克隆自然就是会克隆对象引用的对象了。也就是说,在上例中,改变order1.items并不会影响order2.items了。因为内存中有两个一样的items。

如果实现深克隆?

一个方法自然是重写clone方法,添加如order.items=(LineItems)items.clone()的语句,也就是人为地添加对引用对象的复制。这个方法的缺点是如果引用对象有很多,或者说引用套引用很多重,那么太麻烦了。业界常用的方法是使用串行化然后反串行化的方法来实现深克隆。由于串行化后,对象写到流中,所有引用的对象都包含进来了,所以反串行化后,对等于生成了一个完全克隆的对象。绝!

这个方法的要求是对象(包括被引用对象)必须事先了Serializable接口,否则就要用transient关键字将其排除在复制过程中。

㈡ 如何使Java中的InputStream的深层复制

我们知道在Java中存在这个接口Cloneable,实现该接口的类都会具备被拷贝的能力,同时拷贝是在内存中进行,在性能方面比我们直接通过new生成对象来的快,特别是在大对象的生成上,使得性能的提升非常明显。然而我们知道拷贝分为深拷贝和浅拷贝之分,但是浅拷贝存在对象属性拷贝不彻底问题。关于深拷贝、浅拷贝的请参考这里:渐析java的浅拷贝和深拷贝

一、浅拷贝问题

我们先看如下代码:

[java] view plain
public class Person implements Cloneable{
/** 姓名 **/
private String name;

/** 电子邮件 **/
private Email email;

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Email getEmail() {
return email;
}

public void setEmail(Email email) {
this.email = email;
}

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

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

protected Person clone() {
Person person = null;
try {
person = (Person) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}

return person;
}
}

public class Client {
public static void main(String[] args) {
//写封邮件
Email email = new Email("请参加会议","请与今天12:30到二会议室参加会议...");

Person person1 = new Person("张三",email);

Person person2 = person1.clone();
person2.setName("李四");
Person person3 = person1.clone();
person3.setName("王五");

System.out.println(person1.getName() + "的邮件内容是:" + person1.getEmail().getContent());
System.out.println(person2.getName() + "的邮件内容是:" + person2.getEmail().getContent());
System.out.println(person3.getName() + "的邮件内容是:" + person3.getEmail().getContent());
}
}
--------------------
Output:
张三的邮件内容是:请与今天12:30到二会议室参加会议...
李四的邮件内容是:请与今天12:30到二会议室参加会议...
王五的邮件内容是:请与今天12:30到二会议室参加会议...

在该应用程序中,首先定义一封邮件,然后将该邮件发给张三、李四、王五三个人,由于他们是使用相同的邮件,并且仅有名字不同,所以使用张三该对象类拷贝李四、王五对象然后更改下名字即可。程序一直到这里都没有错,但是如果我们需要张三提前30分钟到,即把邮件的内容修改下:

[java] view plain
public class Client {
public static void main(String[] args) {
//写封邮件
Email email = new Email("请参加会议","请与今天12:30到二会议室参加会议...");

Person person1 = new Person("张三",email);

Person person2 = person1.clone();
person2.setName("李四");
Person person3 = person1.clone();
person3.setName("王五");

person1.getEmail().setContent("请与今天12:00到二会议室参加会议...");

System.out.println(person1.getName() + "的邮件内容是:" + person1.getEmail().getContent());
System.out.println(person2.getName() + "的邮件内容是:" + person2.getEmail().getContent());
System.out.println(person3.getName() + "的邮件内容是:" + person3.getEmail().getContent());
}
}
在这里同样是使用张三该对象实现对李四、王五拷贝,最后将张三的邮件内容改变为:请与今天12:00到二会议室参加会议...。但是结果是:

[java] view plain
张三的邮件内容是:请与今天12:00到二会议室参加会议...
李四的邮件内容是:请与今天12:00到二会议室参加会议...
王五的邮件内容是:请与今天12:00到二会议室参加会议...

这里我们就疑惑了为什么李四和王五的邮件内容也发送了改变呢?让他们提前30分钟到人家会有意见的!

其实出现问题的关键就在于clone()方法上,我们知道该clone()方法是使用Object类的clone()方法,但是该方法存在一个缺陷,它并不会将对象的所有属性全部拷贝过来,而是有选择性的拷贝,基本规则如下:

1、 基本类型

如果变量是基本很类型,则拷贝其值,比如int、float等。

2、 对象

如果变量是一个实例对象,则拷贝其地址引用,也就是说此时新对象与原来对象是公用该实例变量。

3、 String字符串

若变量为String字符串,则拷贝其地址引用。但是在修改时,它会从字符串池中重新生成一个新的字符串,原有紫都城对象保持不变。

基于上面上面的规则,我们很容易发现问题的所在,他们三者公用一个对象,张三修改了该邮件内容,则李四和王五也会修改,所以才会出现上面的情况。对于这种情况我们还是可以解决的,只需要在clone()方法里面新建一个对象,然后张三引用该对象即可:

[java] view plain
protected Person clone() {
Person person = null;
try {
person = (Person) super.clone();
person.setEmail(new Email(person.getEmail().getObject(),person.getEmail().getContent()));
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}

return person;
}

所以:浅拷贝只是Java提供的一种简单的拷贝机制,不便于直接使用。

对于上面的解决方案还是存在一个问题,若我们系统中存在大量的对象是通过拷贝生成的,如果我们每一个类都写一个clone()方法,并将还需要进行深拷贝,新建大量的对象,这个工程是非常大的,这里我们可以利用序列化来实现对象的拷贝。

二、利用序列化实现对象的拷贝

如何利用序列化来完成对象的拷贝呢?在内存中通过字节流的拷贝是比较容易实现的。把母对象写入到一个字节流中,再从字节流中将其读出来,这样就可以创建一个新的对象了,并且该新对象与母对象之间并不存在引用共享的问题,真正实现对象的深拷贝。

[java] view plain
public class CloneUtils {
@SuppressWarnings("unchecked")
public static <T extends Serializable> T clone(T obj){
T cloneObj = null;
try {
//写入字节流
ByteArrayOutputStream out = new ByteArrayOutputStream();
ObjectOutputStream obs = new ObjectOutputStream(out);
obs.writeObject(obj);
obs.close();

//分配内存,写入原始对象,生成新对象
ByteArrayInputStream ios = new ByteArrayInputStream(out.toByteArray());
ObjectInputStream ois = new ObjectInputStream(ios);
//返回生成的新对象
cloneObj = (T) ois.readObject();
ois.close();
} catch (Exception e) {
e.printStackTrace();
}
return cloneObj;
}
}

使用该工具类的对象必须要实现Serializable接口,否则是没有办法实现克隆的。

[html] view plain
public class Person implements Serializable{
private static final long serialVersionUID = 2631590509760908280L;

..................
//去除clone()方法

}

public class Email implements Serializable{
private static final long serialVersionUID = 1267293988171991494L;

....................
}

所以使用该工具类的对象只要实现Serializable接口就可实现对象的克隆,无须继承Cloneable接口实现clone()方法。

㈢ java对象深层复制有什么好处

你说的是不是深拷贝和浅拷贝,深拷贝是把一个原变量的值赋给新的变量,而浅拷贝只是把原变量在计算机的内存地址赋给新的变量,所以深拷贝之后可以独立操作原变量和新变量,互不干扰,而浅拷贝只要新变量或者原变量的值改变了,另一个也改变

㈣ 请问java中深度一个二维数组是什么意思怎么用代码实现

一个二维数组可以看成一个一维数组,每个元素存储一个一维数组首地址的引用,这个没问题吧!
也就是说对于a[][],直接用b[][]=a,或者b[][]=a.clone() 都只是复制了一个引用(包括上面的array等方法),无法保证数据独立性,就是说a数组值改变会影响到b,反之亦然,这就是浅层复制。
如果二维数组存放类型为基本类型,则只需要b的每一行进行复制(Object.clone()可以保证对基本类型做深层复制api上有写):
b[][]=a.clone();//先利用浅层复制分配新的引用存放地址
for(int i=0;i<a.length;i++){
b[i]=a[i].clone();//a[i]指向数组的内容为基本类型,可以深层复制生成新引用对象
}
如果二维数组表示的是引用类型,则要对每一个元素调用clone(),并且保证所表示的引用类型遵循clone()复写原则。
b[][]=a.clone();//先利用浅层复制分配新的引用存放地址
for(int i=0;i<a.length;i++){
for(int j=0;j<a[0].length;j++){
b[i][j]=a[i][j].clone()//为每个元素进行深层复制
}
}

以上是规范写法,实现方法有很多,但一定要记住,单纯的对引用的COPY是没有意义的,编程中要避免。

㈤ java bean 如何过滤某些属性进行深度克隆

序列化bean,先写入,后读出。这两个bean 就毫无关系了。但内容一模一样。

publicSerializableObject(Serializablesobj){
try{
ByteArrayOutputStreambos=newByteArrayOutputStream();

ObjectOutputStreamoos=newObjectOutputStream(bos);

oos.writeObject(sobj);

ByteArrayInputStreambis=newByteArrayInputStream(bos.toByteArray());

ObjectInputStreamois=newObjectInputStream(bis);

return(Serializable)ois.readObject();
}catch(Exceptione){
returnnull;
}

}


可以参考一下。不过 要求实现 Serializable接口。


原文:

http://tieba..com/p/3205322191

㈥ Java中的浅克隆与深克隆 ,有可以区别开的实例吗求大神指点

浅克隆 就是只克隆类的属性,子对象的属性就不管了。

深克隆,就当前类属性、子对象、子子对象属性、、、、、、全克隆了

㈦ java如何深度一个object

方法一:把对象序列化之后再反序列化,之后得到的对象就是深度克隆的对象;
方法二:自己重写方法,不过有点麻烦。

㈧ java中的浅克隆和深克隆是什么

浅克隆:仅仅复制所克隆的对象,而不复制它所引用的对象。

被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象。


深克隆:把要复制的对象所引用的对象都复制了一遍。

那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。


很明显二者时间消耗有差距,在不影响的情况下尽量用浅克隆

注意区分与C语言的浅克隆深克隆 那个是引用地址与重新分配地址构建的区别,详细可以参见:

http://www.cnblogs.com/yxnchinahlj/archive/2010/09/20/1831615.html

㈨ java深度拷贝和浅度拷贝的问题

看来你并没理解错克隆,反而是改变东西的方法不对。
先还是讲讲克隆吧。
是的,上面的clone方法也就是Object的Clone方法是浅表复制,看看他的API说的:
“此方法会创建此对象的类的一个新实例,并像通过分配那样,严格使用此对象相应字段的内容初始化该对象的所有字段;……。所以,此方法执行的是该对象的“浅表复制”,而不“深层复制”操作。 ”
这你没理解错。
但你测试的方法貌似有问题,让人不知你想干什么。
本来sb1、sb2的a字段内容相同的。这两个字段代表两个对象,所谓俩对象,说白了实际上都是一个4字节空间,存的是StringBuffer对象地址,也就是所谓a对象,实际上包含两个东西,一个是a这个指针(我是在说JAVA),另一个指的是这个指针指向的堆栈里面的空间(再复杂就不说了,参阅《JAVA虚拟机》)。相同也就意味着两个变量指向的对象空间一样。
这样的话,如果你调用sb1.a.put...类似的方法朝a中放东西,那sb2也会变。

可你直接把sb1的a换了,也就是让a指向另一个你new的空间。这样和sb2有什么关系,而且你以后再改sb1的a,sb2也不会变。
不说了,累。不懂就问我。

㈩ java,,"浅克隆,只复制一个对象 ,深克隆 对象和引用一起复制"有java实例吗不理解啊

对只包含进本类型的类来说,无所谓深克隆。但是一个类包含集合类或者复杂的类,就要考虑二者的区别了。

class Student {
private int id;
private Address address;
private ArrayList<Book> bookList;
}
对于以上类的一个对象a和克隆类 o来说:
浅克隆:
a !=o 但是 a.address == o.address, a.bookList == o.bookList
深克隆:
a !=o 但是 a.address != o.address, a.bookList != o.bookList
并且 bookList中的每个元素不是相同对象。
简言之, 浅克隆只复制本对象,深克隆除了本对象,其儿子、孙子、....都要克隆。
以上纯手敲,有错请包涵。

热点内容
python导入excel数据 发布:2025-01-11 08:52:49 浏览:569
linux函数脚本 发布:2025-01-11 08:52:49 浏览:827
s4存储卡 发布:2025-01-11 08:48:39 浏览:975
我的世界服务器人数最多的一次 发布:2025-01-11 08:48:37 浏览:325
python音量 发布:2025-01-11 08:48:34 浏览:222
99压缩 发布:2025-01-11 08:43:47 浏览:831
ftp服务器怎么上传 发布:2025-01-11 08:43:45 浏览:518
阅读脚本是什么 发布:2025-01-11 08:39:27 浏览:777
booljava 发布:2025-01-11 08:36:08 浏览:768
我的世界服务器必要弄的东西 发布:2025-01-11 08:32:56 浏览:424