一、Java中的Cloneable接口
Cloneable接口定义在Java.lang包中,它是一个标记接口,并没有什么方法需要实现,但是它的作用却很大,它是用来标记一个类是可以被clone的。
public interface Cloneable { }
这个接口里面一个方法都没有,它只是一个空接口,在Java中类会实现该接口是因为它的存在而已。当标记了一个类实现了Cloneable接口后,该类的clone方法就有了手足无措的支持了。
二、实现Cloneable接口
当要实现一个可以克隆的Java对象时,需要满足三个条件:
- 该类必须实现 Cloneable 接口
- 重写Object类中的clone()方法
- 将protected改为public
下面是一个简单的Car类,它实现了Cloneable接口,重写了clone()方法:
class Car implements Cloneable { String make; String model; int year; Car(String make, String model, int year) { this.make = make; this.model = model; this.year = year; } public Object clone() throws CloneNotSupportedException { return super.clone(); } public String toString() { return "Car{" + "make='" + make + '\'' + ", model='" + model + '\'' + ", year=" + year + '}'; } }
在上面这段代码中,我们重写了Object类的clone()方法。需要注意的是,我们调用的是super.clone(),而不是创建一个新的Car对象。通过使用super.clone(),我们克隆了原始Car对象,并且返回了一个新的对象,这个新的对象是一个Car类型的对象,与原始的Car对象不同。因为这两个Car对象是独立的,一个对象的改变不会影响到另一个对象。
三、Cloneable接口中的问题
Cloneable接口被批评的原因是,当你克隆对象时,该对象的构造函数不会被调用。因此,它的成员变量可能不会被正确地初始化。这就是为什么深拷贝和浅拷贝非常重要,开发人员需要确保对象的所有成员变量都被正确地初始化。
另外,Cloneable接口不是线程安全的。如果在多线程环境下使用clone(),可能会发生竞态条件(race condition)。
四、克隆全局共享不变量对象的风险
在Java中,字符串是全局共享的,所以当你复制一个字符串时,其实是通过引用复制的,而不是克隆。这意味着如果您尝试修改原始字符串中的任何内容,则所有引用该字符串的对象都将受到影响。
下面是一个示例:
public class CloneExample { public static void main(String args[]) throws Exception { String s1 = "hello"; Car car1 = new Car("BMW", "X3", 2021); Car car2 = (Car) car1.clone(); System.out.println("car1 = " + car1); System.out.println("car2 = " + car2); s1 += "world"; System.out.println("s1 = " + s1); } }
在这个例子中,我们创建了一个Car对象并将其克隆,然后我们尝试修改字符串s1。结果,修改仅影响s1变量本身,而不影响任何与s1相关的Car对象。
五、使用Apache Commons Lang实现克隆
Apache Commons Lang是常用的Java工具库之一,其中Apache Commons Lang提供的ObjectUtils类是用于克隆Java对象的强大工具类。ObjectUtils有两个重要的方法,clone()和cloneIfPossible(),可以用来克隆Java对象。
使用ObjectUtils辅助克隆Java对象:
public class CloneExample { public static void main(String args[]) throws Exception { Car car1 = new Car("BMW", "X3", 2021); Car car2 = ObjectUtils.cloneIfPossible(car1); System.out.println("car1 = " + car1); System.out.println("car2 = " + car2); } }
在这个例子中,我们使用了ObjectUtils类的cloneIfPossible()方法,它会根据Java对象是否标记为Cloneable类型,来判断是否需要进行克隆。如果该对象没有被标记为是Cloneable类型,则ObjectUtils将使用Java反射来访问该对象的私有数据成员进行克隆。
六、总结
从多个角度对cloneable接口进行了详细的阐述,包括Java中的Cloneable接口及其实现、Cloneable接口中可能存在的问题、克隆全局共享不变量对象的风险和使用Apache Commons Lang实现克隆等。了解cloneable接口的各个方面,可以帮助开发人员更好地理解如何正确地实现克隆。