Java 对象Clone

本文详细介绍了Java中对象克隆的基本原理与实现步骤。包括如何通过实现Cloneable接口及覆盖clone方法来实现对象的浅层复制与深层复制。通过一个具体的Snake类实例展示了克隆的过程。

一个普通的Java类,继承了Object对象,便继承了该对象的Clone方法

protected native Object clone() throws CloneNotSupportedException;

但是Java基础类或String类却不支持这样的操作

Integer x = new Integer(10);
x.clone(); //编译报错

String s = "abc";
s.clone(); //编译报错

可理解为,Integer类或者String类直接虽然继承了Object,在Integer或者String内部使可以访问clone方法的,由于该方法是protected限制级别。

Java对象Clone对象

     1.实现Cloneable接口。该接口标记类为可克隆的,Object.clone()方法会验证一个类是否实现了Cloneable接口,如果没有,在运行期会throw  CloneNotSupportedException。

     2.可覆盖clone方法,实现语境下的浅层复制或者深层复制。

     3.调用super.clone()方法。

一个简单的例子实现浅层复制

package clone;

/**
 * 克隆研究
 * 
 * @author xl
 *
 */
public class Snake implements Cloneable {

	Snake next;
	char c;

	// 构造器
	public Snake(int i, char x) {
		this.c = x;
		if (--i > 0) {
			next = new Snake(i, (char) (x + 1));
		}
	}

	// 增量
	public void increment() {
		c++;
		if (next != null) {
			next.increment();
		}
	}

	@Override
	public String toString() {
		String s = ":" + c;
		if (next != null) {
			s += next.toString();
		}
		return s;
	}

	@Override
	protected Object clone() throws CloneNotSupportedException {
		return super.clone();
	}

	public static void main(String[] args) throws CloneNotSupportedException {
		Snake s = new Snake(5, 'a');
		System.out.println("s = " + s);
		Snake s2 = (Snake) s.clone();
		System.out.println("s2 = " + s2);
		System.out.println("s = " + s.next.hashCode() + ": " + s.c);
		System.out.println("s2 = " + s2.next.hashCode() + ": " + s2.c);

		//s2.increment();
		System.out.println("after s.increment, s2 = " + s2);
	}

}

再次重载clone,可实现深层复制

@Override
	protected Object clone() throws CloneNotSupportedException {
		Snake s = null;
		s = (Snake) super.clone();
		if (s.next != null)
			s.next = (Snake) s.next.clone();
		return s;
	}
克隆原理

Object.clone()方法负责建立正确的存储容量(原对象的大小),并通过“按位复制”将二进制位从原始对象中复制到新对象的存储空间。所以克隆的第一步应该调用super.clone()方法,为后续的复制建立基础,最后在自己的重载方法中完成必要的操作,以最终完成克隆。

Java 对象克隆时,实体类第一层可以实现深克隆主要是出于数据独立性和数据完整性的考虑。 从数据独立性方面来看,深克隆会创建一个新的对象,新对象的第一层属性会指向全新的内存地址,而不是像浅克隆那样与原对象共享引用类型的属性。对于引用类型的成员变量,浅克隆只是复制引用,多个对象会指向同一个内存地址,修改其中一个对象的属性会影响到其他共享该引用的对象。而深克隆则为每个引用类型的属性创建新的实例,使得新对象和原对象在第一层属性上相互独立,互不影响。例如,在涉及多个对象操作的场景中,如果不进行深克隆,一个对象对共享引用属性的修改可能会导致其他对象出现意外的结果,而深克隆可以避免这种情况的发生,保证每个对象的数据独立性 [^1]。 从数据完整性方面来说,当实体类的第一层属性包含复杂的引用类型时,深克隆可以确保新对象完整地复制原对象的状态。如果只是浅克隆,可能会因为共享引用而导致数据不一致。比如一个包含多个属性的实体类,其中某个属性是自定义的类对象,深克隆会将这个自定义类对象也进行复制,使得新对象拥有完整的数据副本,保证了数据的完整性 [^1]。 另外,对于基本数据类型和不可变类(如 String 和基本类型的包装类),由于它们本身的特性,在复制时不会出现共享引用导致的数据问题。基本数据类型直接存储值,复制时会复制其值;不可变类每次操作都会创建新的对象,所以不用担心修改后会影响克隆对象的属性值,这也为实体类第一层实现深克隆提供了便利 [^4]。 ```java import java.util.Objects; // 假设这是一个实体类 class Entity { private int basicType; // 基本数据类型 private String immutableType; // 不可变类 private InnerClass referenceType; // 引用类型 public Entity(int basicType, String immutableType, InnerClass referenceType) { this.basicType = basicType; this.immutableType = immutableType; this.referenceType = referenceType; } // 深克隆方法 public Entity deepClone() { InnerClass clonedInner = null; if (this.referenceType != null) { clonedInner = this.referenceType.clone(); } return new Entity(this.basicType, this.immutableType, clonedInner); } // Getters and Setters public int getBasicType() { return basicType; } public void setBasicType(int basicType) { this.basicType = basicType; } public String getImmutableType() { return immutableType; } public void setImmutableType(String immutableType) { this.immutableType = immutableType; } public InnerClass getReferenceType() { return referenceType; } public void setReferenceType(InnerClass referenceType) { this.referenceType = referenceType; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; Entity entity = (Entity) o; return basicType == entity.basicType && Objects.equals(immutableType, entity.immutableType) && Objects.equals(referenceType, entity.referenceType); } @Override public int hashCode() { return Objects.hash(basicType, immutableType, referenceType); } } // 内部类,需要支持克隆 class InnerClass implements Cloneable { private String innerProperty; public InnerClass(String innerProperty) { this.innerProperty = innerProperty; } @Override public InnerClass clone() { try { return (InnerClass) super.clone(); } catch (CloneNotSupportedException e) { throw new AssertionError(); } } // Getters and Setters public String getInnerProperty() { return innerProperty; } public void setInnerProperty(String innerProperty) { this.innerProperty = innerProperty; } @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; InnerClass that = (InnerClass) o; return Objects.equals(innerProperty, that.innerProperty); } @Override public int hashCode() { return Objects.hash(innerProperty); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值