创建型模式之原型模式
定义与特点
原型(Prototype)模式的定义如下:用一个已经创建的实例作为原型,通过复制该原型对象来创建一个和原型相同或相似的新对象。
在这里,原型实例指定了要创建的对象的种类。用这种方式创建对象非常高效,根本无须知道对象创建的细节。
例如,Windows 操作系统的安装通常较耗时,如果复制就快了很多。在生活中复制的例子非常多,这里不一一列举了。
结构与实现
由于 C# 提供了 ICloneable 接口,用 C# 实现原型模式很简单。
模式的结构
原型模式包含以下主要角色:
- 原型(Prototype):声明一个克隆自身的接口,该角色一般有抽象类(Prototype)、接口(ICloneable)两种实现方式(GOF使用的是抽象类,在C#中个人推荐使用接口)。
- 具体原型类(ConcretePrototype):实现原型(抽象类或接口)的 Clone() 方法,它是可被复制的对象。
- 访问类(Client):使用具体原型类中的 Clone() 方法来复制新的对象。
使用接口作为Prototype时,其结构图如图所示(后面全部是接口方式的原型模式实现):
使用抽象类作为Prototype时,其结构图如图所示(来自GOF):
模式的实现
C# 中已经定义了 ICloneable 接口,具体原型类只要实现 ICloneable 接口就可实现对象的克隆(Object有 MemberwiseClone() 方法默认浅克隆),克隆是浅克隆还是深克隆取决于具体原型类 Clone() 的实现。其代码如下:
//原型接口:该接口不需要自己定义,C#中已经定义好了
public interface ICloneable
{
Object Clone();
}
//抽象原型类
public abstract class Prototype
{
public abstract Object Clone();
}
//具体原型类(接口方式)
public class ConcretePrototype : /*Prototype,*/ICloneable
{
private int id;
public int Id
{
get { return id; }
}
public ConcretePrototype(int id)
{
this.id = id;
}
public Object Clone()
{
return new ConcretePrototype(id);
}
//使用抽象原型类
//public override Object Clone()
//{
// //Object的默认克隆方式(浅克隆)
// return (Prototype)this.MemberwiseClone();
//}
}
//访问类,这里直接用的控制台的Program类
class Program
{
static void Main(string[] args)
{
ConcretePrototype cp1 = new ConcretePrototype(1);
ConcretePrototype cp2 = (ConcretePrototype)cp1.Clone();
Console.WriteLine("cp1的Id为:{0}",cp1.Id);
Console.WriteLine("cp2的Id为:{0}", cp2.Id);
Console.ReadKey();
}
}
运行结果:
cp1的Id为:1
cp2的Id为:1
应用场景
原型模式通常适用于以下场景:
- 对象之间相同或相似,即只是个别的几个属性不同的时候。
- 对象的创建过程比较麻烦,但复制比较简单的时候。
扩展:带原型管理器的原型模式
原型模式可扩展为带原型管理器的原型模式,它在原型模式的基础上增加了一个原型管理器 PrototypeManager 类。该类用 Dictionary(或HashMap) 保存多个复制的原型,Client 类可以通过管理器的 Get(String id) 方法从中获取复制的原型。其结构如图所示: