java.请写出一个原型模式_Java设计模式-原型模式

原型模式(Prototype)

原型模式属于对象的创建模式。通过给出一个原型对象来指明所有创建的对象的类型,然后用复制这个原型对象的办法创建更多同类型的对象。这就是原型模式的用意。

原型模式的结构

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

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

简单形式的原型模式

364fe60905f24d3ea0f4b28ec36c2f87.png

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

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

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

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

源代码

抽象原型角色:

packagecom.tutorialspoint;public interfacePrototype {/** 克隆自身的方法

**/

publicPrototype clone();

}

具体原型角色:

packagecom.tutorialspoint;public class ConcretePrototype1 implementsPrototype {

@OverridepublicPrototype clone() {//TODO Auto-generated method stub//return super.clone();

/** 最简单地克隆,新建一个自身对象,由于没有属性就不再复制值了

**/Prototype prototype= newConcretePrototype1();returnprototype;

}

}

packagecom.tutorialspoint;public class ConcretePrototype2 implementsPrototype {

@OverridepublicPrototype clone() {//TODO Auto-generated method stub

Prototype prototype = newConcretePrototype2();returnprototype;

}

}

客户端角色:

packagecom.tutorialspoint;public classClient {//持有需要使用的原型接口对象

privatePrototype prototype;/** 构造方法,传入需要使用的原型接口对象

**/

publicClient(Prototype prototype) {this.prototype =prototype;

}public voidoperation(Prototype example) {//需要创建原型接口的对象

Prototype copyPrototype =prototype.clone();

}

}

登记形式的原型模式

166be9f497db8346fbdd2833a4125ac1.png

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

源代码

抽象原型角色:

packagecom.tutorialspoint;public interfacePrototype {/** 克隆自身的方法

**/

publicString getName();public voidsetName(String name);publicPrototype clone();

}

具体原型角色:

packagecom.tutorialspoint;public class ConcretePrototype1 implementsPrototype {privateString name;

@OverridepublicPrototype clone() {//TODO Auto-generated method stub//return super.clone();

/** 最简单地克隆,新建一个自身对象,由于没有属性就不再复制值了

**/Prototype prototype= newConcretePrototype1();returnprototype;

}

@OverridepublicString getName() {//TODO Auto-generated method stub

returnname;

}

@Overridepublic voidsetName(String name) {//TODO Auto-generated method stub

this.name =name;

}

}packagecom.tutorialspoint;public class ConcretePrototype2 implementsPrototype {privateString name;

@OverridepublicPrototype clone() {//TODO Auto-generated method stub

Prototype prototype = newConcretePrototype2();returnprototype;

}

@OverridepublicString getName() {//TODO Auto-generated method stub

returnname;

}

@Overridepublic voidsetName(String name) {//TODO Auto-generated method stub

this.name =name;

}

}

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

packagecom.tutorialspoint;importjava.util.Map;importjava.util.HashMap;public classPrototypeManager {/** 用来记录原型的编号和原型实例的对应关系

**/

private static Map map = new HashMap<>();/** 私有化构造方法,避免外部创建实例

*

**/

privatePrototypeManager() {/**向原型管理器里面添加或是修改某个原型的注册

**/}public synchronized static voidsetPrototype(String prototypeId , Prototype prototype) {/** 向原型管理器里面添加或是修改某个原型注册

* @param prototypeId 原型编号

* @param prototype 原型实例

**/map.put(prototypeId, prototype);

}public synchronized static voidremovePrototype(String prototypeId) {

map.remove(prototypeId);

}public synchronized static Prototype getPrototype(String prototypeId) throwsException {

Prototype prototype=map.get(prototypeId);if ( prototype == null) {throw new Exception("你希望获取的原型还没有注册或者已被注销");

}returnprototype;

}

}

客户端角色:

public static voidmain( String[] args ) {try{

Prototype p1= newConcretePrototype1();

PrototypeManager.setPrototype("p1", p1);//获取原型来创建对象

Prototype p3 = PrototypeManager.getPrototype("p1").clone();

p3.setName("张三");

System.out.println("第一个实例: " +p3);//有人动态的切换了实现

Prototype p2 = newConcretePrototype2();

PrototypeManager.setPrototype("p2", p2);//重新获取原型来创建对象

Prototype p4 = PrototypeManager.getPrototype("p2").clone();

p4.setName("李四");

System.out.println("第二个实例: " +p4);//有人注销了这个原型

PrototypeManager.removePrototype("p1");//有人再次获取原型来创建对象

Prototype p5 = PrototypeManager.getPrototype("p1").clone();

p5.setName("王五");

System.out.println("第三个实例: " +p5);

}catch(Exception e) {

e.printStackTrace();

}

}

执行结果:

c77ecd5f3988618bca15c58d93912462.png

两种形式的比较

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

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

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

原型模式的优点

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

原型模式的缺点

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

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值