[coolxing按: 转载请注明作者和出处, 如有谬误, 欢迎在评论中指正.]
clone用在什么场合下
1. 创建一个新的对象需要很复杂的步骤(例如类的继承层次很深), 而刚好有一个适合的对象用来clone,
这时使用clone技术有助于性能优化;
2. 用在原型(prototype)模式下.
shallow copy和deep copy
两者本质上的区别在于, 采用shallow copy克隆得到的对象和被克隆对象的引用属性指向相同的地址. 而采用deep copy克隆得到的对象和被克隆对象的引用属性指向不同的地址.
采用shallow copy的方式克隆一个对象时, 如果对象中包含引用属性, 那么直接复制一份引用给新对象的对应属性. 而deep copy则会为新对象的引用属性开辟新的内存空间然后再将地址赋值给引用.
这里的”引用属性”泛指除基本数据类型, 包装数据类型, String类型之外的属性.
Object类的clone方法是一个native方法, 属于shallow
copy的范畴. 调用该方法时首先会判断类是否实现了Cloneable接口, 如果没有实现Cloneable接口, 会抛出CloneNotSupportedException异常.
调用Object类的clone方法实现shallow copy的例子:
public class Person implements Cloneable {
private int age;
private String name;
private Address address;
public Person(int age, String name, Address address) {
this.age = age;
this.name = name;
this.address = address;
}
@Override
public Person clone() throws CloneNotSupportedException {
// 直接调用Object类的clone方法实现的是shallow copy
return (Person) super.clone();
}
public static void main(String[] args) {
Person old = new Person(24, "coolxing", new Address("beijing"));
try {
Person newPerson = old.clone();
System.out.println("没有更改newPerson的属性时, newPerson.name = "
+ newPerson.name + ", newPerson.age = " + newPerson.age
+ ", newPerson.address.getCity() = "
+ newPerson.address.getCity());
System.out.println("没有更改newPerson的属性时, old.name = " + old.name
+ ", old.age = " + old.age + ", old.address.getCity() = "
+ old.address.getCity());
newPerson.name = "min";
newPerson.age = 22;
newPerson.address.setCity("dalian");
System.out.println("更改了newPerson的属性时, newPerson.name = "
+ newPerson.name + ", newPerson.age = " + newPerson.age
+ ", newPerson.address.getCity() = "
+ newPerson.address.getCity());
System.out.println("更改了newPerson的属性时, old.name = " + old.name
+ ", old.age = " + old.age + ", old.address.getCity() = "
+ old.address.getCity());
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
}
}
public class Address {
private String city;
public Address(String city) {
this.city = city;
}
public String getCity() {
return city;
}
public void setCity(String city) {
this.city = city;
}
}
程序输出:
没有更改newPerson的属性时, newPerson.name = coolxing,
newPerson.age = 24, newPerson.address.getCity() = beijing
没有更改newPerson的属性时, old.name = coolxing, old.age = 24,
old.address.getCity() = beijing
更改了newPerson的属性时, newPerson.name = min,
newPerson.age = 22, newPerson.address.getCity() = dalian
更改了newPerson的属性时, old.name = coolxing, old.age = 24,
old.address.getCity() = dalian
由此可见, old对象和newPerson对象的address属性指向同一个Address对象. 更改newPerson对象的address属性的值会影响到old对象.
怎样实现deep copy
shallow copy并非一无是处, 也有相应的应用场合. 如果确实是需要deep
copy, 那就只能由开发者自己实现. 通常实现deep copy的方式有2种:
1. 递归调用引用属性对象的clone方法.
为Person类增加新的方法:
public Person deepClone() throws CloneNotSupportedException {
Person newPerson = clone();
newPerson.address = address.clone();
return newPerson;
}
这需要Address类也实现Cloneable接口并覆写clone方法:
public Address clone() throws CloneNotSupportedException {
return (Address) super.clone();
}
在测试方法中调用deepClone方法代替clone方法, 就会得到输出:
更改了newPerson的属性时, newPerson.name = min,
newPerson.age = 22, newPerson.address.getCity() = dalian
更改了newPerson的属性时, old.name = coolxing, old.age = 24,
old.address.getCity() = beijing
由此可见正确实现了deep copy, 对newPerson对象属性的修改不会再影响到old对象.
2. 反序列化. 先将对象序列化, 然后再反序列化得到原有对象的拷贝, 这是非常彻底的deep copy. 前提是类实现了Serializable接口.
增加deepCloneSerializable方法:
public Person deepCloneSerializable() throws IOException, ClassNotFoundException {
ByteArrayOutputStream buffer = new ByteArrayOutputStream();
ObjectOutputStream obOut = new ObjectOutputStream(buffer);
obOut.writeObject(this);
obOut.flush();
ObjectInputStream obIn = new ObjectInputStream(
new ByteArrayInputStream(buffer.toByteArray()));
return (Person) obIn.readObject();
}
这需要Address类也实现Serializable接口. 在测试代码中调用deepCloneSerializable方法, 输出结果同上.
分享到:
相关推荐
用 Java 语言编写的 初学者可学习的clone()
Java Clone(类的复制)实例代码 Java Clone(类的复制)实例代码
详细的描述了Java中 clone方法使用
Clone 属性的相关内容 Clone 属性的相关内容
今天小编就为大家分享一篇关于Java Clone深拷贝与浅拷贝的两种实现方法,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
主要介绍了 Java clone方法详解及简单实例的相关资料,需要的朋友可以参考下
java中clone的详细用法,分浅拷贝和深拷贝,并分别有详细的实例介绍。从原理分析。
Java中的clone方法详解_动力节点Java学院,动力节点口口相传的Java黄埔军校
Java中的克隆(Clone)机制,很少涉及的内容。
NULL 博文链接:https://gaoquanyang.iteye.com/blog/1177183
Java深浅clone测试代码 流拷贝 Cloneable
clone顾名思义就是复制, 在Java语言中, clone方法被对象调用,所以会复制对象。下面通过本文给大家介绍java中的clone方法,感兴趣的朋友一起看看吧
主要介绍了java 中clone()的使用方法的相关资料,希望通过本文能帮助大家能掌握clone()的克隆方法,需要的朋友可以参考下
java中仅有的创建对象的两种方式:①.使用new操作符创建对象;②....下面这篇文章主要通过JVM角度给大家详细谈谈Java的clone操作的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下
clone的用法 希望有帮助,仅供参考 通过例子的分析,可以对克隆的方法有所深入了解
java code clone
并且由于Java不能通过简单的赋值来解决对象复制的问题,在开发过程中,也常常要要应用clone()方法来复制对象。比如函数参数类型是自定义的类时,此时便是引用传递而不是值传递。以下是一个小例子: ...
java中clone的详细用法,分浅拷贝和深拷贝,并分别有详细的实例介绍。从原理分析。