java clone

clone是什么?

clone()是java提供一个功能,用于获取对象的一个副本。对于副本的的心理预期(并非必须)应该是[^1]:

  • x.clone() != x 因为目标是对象,拷贝与原对象不应该是同一对象。
  • x.clone().getClass() == x.getClass() 他们具有相同的类。
  • x.clone().equals(x)==true 在比较时,它们应该是相等的。

clone()实质就是java就是对原型模式的原生支持

clone()的实现

语法要求

Java提供了clone()方法的默认实现,在Object类中进行了声明:

protected native Object clone() throws CloneNotSupportedException;

首先可以看到它是native方法。Object.clone()执行时,会创建该类的新实例,并对所有字段进行浅拷贝: 基本类型字段复制值,对于引用类型字段只复制引用。

其次它是protected方法,要求子类如果需要提供该功能,必须覆盖该方法,并声明为public。

@override
public Type clone(){...}

最后可以看到其抛出CloneNotSupportedException异常,其要求子类如果提供了该功能,而且使用了Java默认实现(即调用了super.clone()),必须声明Cloneable标记接口

虽然不使用默认实现时,不声明Cloneable也不会报错,但只要提供了clone()功能,就最好声明实现Cloneable。

public MyClass implements Cloneable{}

实现细节

参考Joshua Bloch的文章[^2][^4],实际要实现一个功能良好的clone()方法要考虑很多方面,而且存在其它更清晰的方式替代(如复制构造器)。

大概总结一下该文章(你最好亲自去看),就是:

  • java语言对clone功能的设计是失败的。
  • 合理的clone()内部实现,应该是一个clone()链条,通过层层super.clone(),最终到达Object.clone()。“类似构造器调用链”。
  • 每一层都应该处理好实例域,包括需要更改的基本类型数据和需要深拷贝的引用类型数据。
  • 要考虑线程安全、final字段的不可修改、异常处理等问题。

clone的应用

因为上述原因(Josua Bloch的文章),“应该尽量避免使用clone,clone显得有些笨拙”[^3]。从大神们的观点来看,除了数组的clone(),不要在其它地方使用clone[^4]。

实例场景

我对Spring Framework 的v4.0.2.RELEASE版本为样本,统计了Java代码实际应用中clone出现的次数:

正式代码

类型次数
java.util.Stack1
array14
WebClient.Builder1
WebHttpHandlerBuilder1

测试代码

类型次数
java.text.SimpleDateFormat1
java.util.LinkedHashMap2
LinkedCaseInsensitiveMap1
UriComponentsBuilder1

主要使用于在array和ADT这类数据集合,且很少被使用。

[^1] javadoc: Object.clone()
[^2] 《EffectiveJava》11谨慎覆盖clone
[^3]《java核心技术 卷1》6.2 对象克隆
[^4]Copy Constructor versus Cloning

转载于:https://www.cnblogs.com/redreampt/p/9418960.html

Java中,对象克隆可以通过两种方式实现:浅克隆和深克隆。 浅克隆是指创建一个新的对象,该对象的字段与原始对象相同。但如果字段是引用类型,那么新对象与原始对象将引用相同的引用类型对象。换句话说,新对象和原始对象共享相同的引用类型字段,对其中一个对象所做的更改会影响另一个对象。 要实现浅克隆,需要在要克隆的类中实现Cloneable接口,并重写Object类的clone()方法。以下是一个示例: ```java public class MyClass implements Cloneable { private int myField; public Object clone() throws CloneNotSupportedException { return super.clone(); } // 其他代码... } ``` 使用浅克隆时,可以通过调用clone()方法来创建一个新对象: ```java MyClass original = new MyClass(); MyClass clone = (MyClass) original.clone(); ``` 深克隆是指创建一个新的对象,并且递归地克隆其所有字段,包括引用类型字段。这样,新对象与原始对象完全独立,对其中一个对象所做的更改不会影响另一个对象。 要实现深克隆,需要在要克隆的类中实现Serializable接口,并使用序列化和反序列化来实现深复制。以下是一个示例: ```java import java.io.*; public class MyClass implements Serializable { private int myField; private MyReferenceType reference; public Object deepClone() throws IOException, ClassNotFoundException { ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream out = new ObjectOutputStream(bos); out.writeObject(this); ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream in = new ObjectInputStream(bis); return in.readObject(); } // 其他代码... } ``` 使用深克隆时,可以通过调用deepClone()方法来创建一个新对象: ```java MyClass original = new MyClass(); MyClass clone = (MyClass) original.deepClone(); ``` 需要注意的是,深克隆要求对象及其引用类型字段都必须实现Serializable接口。而且,深克隆可能会影响性能,因为它涉及序列化和反序列化的过程。因此,在使用克隆时需要谨慎考虑其适用性和性能影响。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值