原型模式:通过复制一个已经存在的实例来返回新的实例,而不是新建实例.被复制的实例就是我们所称的原型,这个原型是可定制的(clone)
特点:
- 向用户隐藏了创建新实例的复杂性
- 让用户可以产生未知类型的对象,即只需知道对象实现了哪些接口,而无需知道具体是哪种实现
- 在某些情况下,复制对象比创建对象更高效
Prototype类需要具备以下两个条件:
- 实现Cloneable接口。在java语言有一个Cloneable接口,它的作用只有一个,就是在运行时通知虚拟机可以安全地在实现了此接口的类上使用clone方法。在java虚拟机中,只有实现了这个接口的类才可以被拷贝,否则在运行时会抛出CloneNotSupportedException异常。
- 重写Object类中的clone方法。Java中,所有类的父类都是Object类,Object类中有一个clone方法,作用是返回对象的一个拷贝,但是其作用域protected类型的,一般的类无法调用,因此,Prototype类需要将clone方法的作用域修改为public类型。
注意复制对象时的浅拷贝与深拷贝:
- 浅拷贝: 对值类型的成员变量进行值的复制,对引用类型的成员变量只复制引用,不复制引用的对象
- 深拷贝: 对值类型的成员变量进行值的复制,对引用类型的成员变量也进行引用对象的复制
注意:
- 使用原型模式复制对象不会调用类的构造方法。因为对象的复制是通过调用Object类的clone方法来完成的,它直接在内存中复制数据,因此不会调用到类的构造方法
- 单例模式与原型模式是冲突的,在使用时要特别注意
实现深拷贝的克隆:
1 public class Prototype implements Cloneable { 2 private ArrayList list = new ArrayList(); 3 public Prototype clone(){ 4 Prototype prototype = null; 5 try{ 6 prototype = (Prototype)super.clone(); 7 prototype.list = (ArrayList) this.list.clone(); 8 }catch(CloneNotSupportedException e){ 9 e.printStackTrace(); 10 } 11 return prototype; 12 } 13 }