【Java设计模式】之原型模式

1.定义

原型模式(Prototype),用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。

原型模式要求对象实现一个可以“克隆”自身的接口,这样就可以通过复制一个实例对象本身来创建一个新的实例。这样一来,通过原型实例创建新的对象,就不再需要关心这个实例本身的类型,只要实现了克隆自身的方法,就可以通过这个方法来获取新的对象,而无须再去通过new来创建。

原型模式有两种表现形式:(1)简单形式、(2)登记形式,这两种表现形式仅仅是原型模式的不同实现。

2.简单形式的原型模式类图

这种形式涉及到三个角色:

1)客户(Client)角色:客户类提出创建对象的请求。

2)抽象原型(Prototype)角色:这是一个抽象角色,通常由一个Java接口或Java抽象类实现。此角色给出所有的具体原型类所需的接口。

3)具体原型(Concrete Prototype)角色:被复制的对象。此角色需要实现抽象的原型角色所要求的接口。

3.简单形式的原型模式代码

抽象原型角色

public interface Prototype {

	/**
	 * 克隆自身的方法
	 * @return
	 */
	public Object clone();
}

具体原型角色

public class ConcretePrototype1 implements Prototype {
	public Object clone() {
		// 最简单的克隆,新建一个自身对象,由于没有属性就不再复制值了
		Prototype prototype = new ConcretePrototype1();
		return prototype;
	}
}

public class ConcretePrototype2 implements Prototype {
	public Object clone() {
		// 最简单的克隆,新建一个自身对象,由于没有属性就不再复制值了
		Prototype prototype = new ConcretePrototype2();
		return prototype;
	}
}

客户端角色

public class Client {
	private Prototype prototype;
	public Client(Prototype prototype) {
		super();
		this.prototype = prototype;
	}
	public void operation(){
		Prototype copyPrototype = (Prototype) prototype.clone();
		System.out.println("copyPrototype="+copyPrototype);
	}
}

4.登记形式的原型模式类图


作为原型模式的第二种形式,它多了一个原型管理器(PrototypeManager)角色,该角色的作用是:创建具体原型类的对象,并记录每一个被创建的对象。

5.登记形式的原型模式代码

抽象原型角色

public interface Prototype {

	/**
	 * 克隆自身的方法
	 * @return
	 */
	public Object clone();
	public String getName();
	public void setName(String name);
}

具体原型角色

public class ConcretePrototype1 implements Prototype {
	private String name;
	
	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	public String toString() {
		return "ConcretePrototype1 [name=" + name + "]";
	}

	public Object clone() {
		// 最简单的克隆,新建一个自身对象,由于没有属性就不再复制值了
		Prototype prototype = new ConcretePrototype1();
		prototype.setName(name);
		return prototype;
	}
}

原型管理器角色保持一个聚集,作为对所有原型对象的登记,这个角色提供必要的方法,供外界增加新的原型对象和取得已经登记过的原型对象。

public class PrototypeManager {
	private static Map<String, Prototype> map = new HashMap<String, Prototype>();

	private PrototypeManager() {
	}

	/**
	 * 向原型管理器里面添加或是修改某个原型注册
	 * 
	 * @param prototypeId
	 *            原型编号
	 * @param prototype
	 *            原型实例
	 */
	public synchronized static void setPrototype(String prototypeId,
			Prototype prototype) {
		map.put(prototypeId, prototype);
	}

	/**
	 * 从原型管理器里面删除某个原型注册
	 * 
	 * @param prototypeId
	 *            原型编号
	 */
	public synchronized static void removePrototype(String prototypeId) {
		map.remove(prototypeId);
	}

	/**
	 * 获取某个原型编号对应的原型实例
	 * 
	 * @param prototypeId
	 *            原型编号
	 * @return 原型编号对应的原型实例
	 * @throws Exception
	 *             如果原型编号对应的实例不存在,则抛出异常
	 */
	public synchronized static Prototype getPrototype(String prototypeId)
			throws Exception {
		Prototype prototype = map.get(prototypeId);
		if (prototype == null) {
			throw new Exception("您希望获取的原型还没有注册或已被销毁");
		}
		return prototype;
	}
}

客户端角色

public static void main(String[] args) {
		try {
			Prototype p1 = new ConcretePrototype1();
			PrototypeManager.setPrototype("p1", p1);
			// 获取原型来创建对象
			Prototype p3 = (Prototype) PrototypeManager.getPrototype("p1").clone();
			p3.setName("张三");
			System.out.println("第一个实例:" + p3);
			// 有人动态的切换了实现
			Prototype p2 = new ConcretePrototype2();
			PrototypeManager.setPrototype("p1", p2);
			// 重新获取原型来创建对象
			Prototype p4 = (Prototype) PrototypeManager.getPrototype("p1").clone();
			p4.setName("李四");
			System.out.println("第二个实例:" + p4);
			// 有人注销了这个原型
			PrototypeManager.removePrototype("p1");
			// 再次获取原型来创建对象
			Prototype p5 = (Prototype) PrototypeManager.getPrototype("p1").clone();
			p5.setName("王五");
			System.out.println("第三个实例:" + p5);
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

6.两种形式的比较

简单形式和登记形式的原型模式各有其长处和短处。

如果需要创建的原型对象数目较少而且比较固定的话,可以采取第一种形式。在这种情况下,原型对象的引用可以由客户端自己保存。

如果要创建的原型对象数目不固定的话,可以采取第二种形式。在这种情况下,客户端不保存对原型对象的引用,这个任务被交给管理员对象。在复制一个原型对象之前,客户端可以查看管理员对象是否已经有一个满足要求的原型对象。如果有,可以直接从管理员类取得这个对象引用;如果没有,客户端就需要自行复制此原型对象。

7.Java中的克隆方法

Java的所有类都是从java.lang.Object类继承而来的,而Object类提供protected Object clone()方法对对象进行复制,子类当然也可以把这个方法置换掉,提供满足自己需要的复制方法。对象的复制有一个基本问题,就是对象通常都有对其他的对象的引用。当使用Object类的clone()方法来复制一个对象时,此对象对其他对象的引用也同时会被复制一份

Java语言提供的Cloneable接口只起一个作用,就是在运行时期通知Java虚拟机可以安全地在这个类上使用clone()方法。通过调用这个clone()方法可以得到一个对象的复制。由于Object类本身并不实现Cloneable接口,因此如果所考虑的类没有实现Cloneable接口时,调用clone()方法会抛出CloneNotSupportedException异常。

8.克隆满足的条件

clone()方法将对象复制了一份并返还给调用者。所谓“复制”的含义与clone()方法是怎么实现的。一般而言,clone()方法满足以下的描述:

1)对任何的对象x,都有:x.clone()!=x。换言之,克隆对象与原对象不是同一个对象。

2)对任何的对象x,都有:x.clone().getClass() == x.getClass(),换言之,克隆对象与原对象的类型一样。

3)如果对象xequals()方法定义其恰当的话,那么x.clone().equals(x)应当成立的。

JAVA语言的API中,凡是提供了clone()方法的类,都满足上面的这些条件。JAVA语言的设计师在设计自己的clone()方法时,也应当遵守着三个条件。一般来说,上面的三个条件中的前两个是必需的,而第三个是可选的。

9.浅克隆和深克隆

无论你是自己实现克隆方法,还是采用Java提供的克隆方法,都存在一个浅度克隆和深度克隆的问题。

浅度克隆

只负责克隆按值传递的数据(比如基本数据类型、String类型),而不复制它所引用的对象,换言之,所有的对其他对象的引用都仍然指向原来的对象。

深度克隆

除了浅度克隆要克隆的值外,还负责克隆引用类型的数据。那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。换言之,深度克隆把要复制的对象所引用的对象都复制了一遍,而这种对被引用到的对象的复制叫做间接复制。

深度克隆要深入到多少层,是一个不易确定的问题。在决定以深度克隆的方式复制一个对象的时候,必须决定对间接复制的对象时采取浅度克隆还是继续采用深度克隆。因此,在采取深度克隆时,需要决定多深才算深。此外,在深度克隆的过程中,很可能会出现循环引用的问题,必须小心处理。

10.原型模式的优缺点

原型模式的优点

原型模式允许在运行时动态改变具体的实现类型。原型模式可以在运行期间,由客户来注册符合原型接口的实现类型,也可以动态地改变具体的实现类型,看起来接口没有任何变化,但其实运行的已经是另外一个类实例了。因为克隆一个原型就类似于实例化一个类。

原型模式的缺点

原型模式最主要的缺点是每一个类都必须配备一个克隆方法。配备克隆方法需要对类的功能进行通盘考虑,这对于全新的类来说不是很难,而对于已经有的类不一定很容易,特别是当一个类引用不支持序列化的间接对象,或者引用含有循环结构的时候。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值