原型模式:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。(要创建一个对象,这个对象为实现原型接口,方法是原型克隆。克隆只是方法而不是原型模式的目的,创建对象才是目的)

UML类图:

构成:

1.原型接口,提供给客户操作,声明一个clone()方法,用于克隆自身创建另一个实例

2.原型,实现clone()方法,分为浅层克隆和深层克隆。浅层克隆,值类型成员的值相同,但是在内存是存储在不同空间上的,而引用类型成员是相同的引用,即是同一块内存空间,修改其中一个,另一个也随之改变。而深层克隆,不论是值类型还是引用类型,都是不同的内存。

以电脑为例,先定义电脑接口,包含clone()方法

 interface Computer
{
string CPU { get; set; }
string Memory { get; set; }
//string GPU { get; set; }
GPU GPU { get; set; }
Computer clone();
Computer deepClone();
}

原型实现这个接口,因为.NET所有类都是object的子类,而object类有一个MemberwiseClone()方法返回自身的浅层克隆,所以可以用它实现浅层克隆clone()(自己实现?)。而深层克隆deepClone()是用序列化和反序列化实现的(这个现在还不懂为什么)

class ComputerPrototype : Computer
{
private string cpu;
private string memory;
private GPU gpu;
public Computer clone()
{
return (Computer)this.MemberwiseClone();
}
public Computer deepClone()
{
MemoryStream stream = new MemoryStream();
BinaryFormatter formatter = new BinaryFormatter();
formatter.Serialize(stream, this);
stream.Position = ;
return formatter.Deserialize(stream) as Computer;
}
public override string ToString()
{
return CPU + "\n" + Memory + "\n" + GPU.Producer+" "+gpu.Type;
}
public GPU GPU
{
get { return gpu; }
set { gpu = value; }
}
public string Memory
{
get { return memory; }
set { memory = value; }
}
public string CPU
{
get { return cpu; }
set { cpu = value; }
}
}

Computer包含一个类GPU,用来演示浅层和深层克隆的不同,它被标记为可序列化

[Serializable]
class GPU
{
public GPU(string producer, string type)
{
this.Producer = producer;
this.Type = type;
}
public string Producer
{
get;
set;
}
public string Type
{
get;
set;
}
}

先实例化一个ComputerPrototype原型,再为它的成员赋值,接下来,通过浅层克隆方法clone()创建一个实例,修改新实例的成员,最后查看两个实例的异同

class Program
{
static void Main(string[] args)
{
Computer low = new ComputerPrototype();
low.CPU = "e3 1231v3";
low.Memory = "4 G";
low.GPU = new GPU("凄惨红", "GTX970");
//Computer high = low.deepClone();
Computer high = low.clone(); high.GPU.Producer = "高贵的阿苏斯";
high.Memory = "8 G";
Console.Write("GPU是否是同一引用:");
Console.WriteLine(low.GPU == high.GPU); Console.WriteLine("Low:");
Console.WriteLine(low.ToString());
Console.WriteLine("High:");
Console.WriteLine(high.ToString());
Console.ReadKey();
}
}

结果如下

结果表明,两个实例的GPU是同一引用,修改了克隆,原型也被修改了,但是Memory确是各自独立的,修改克隆并不影响原型。

而将clone()改为执行deepClone()方法结果为

这表明,两个实例GPU为不同引用,修克隆的GPU和Memory不会影响原型。

效果:

1.运行时刻增加和删除产品。运行时注册原型就可以增加新的产品(low是一个新产品)

2.改变值以指定新对象。减少需要的类的数量(通过改变一些值,创建无数种对象,high为一个新对象)

3.改变结构以指定新对象。(?)

4.减少子类的构造。工厂方法,一个产品一个ConcreteCreator,而原型模式不需要

5.用类动态配置应用。

设计模式C#实现(八)——原型模式的更多相关文章

  1. 设计模式(六)原型模式(Prototype Pattern)

    一.引言 在软件系统中,当创建一个类的实例的过程很昂贵或很复杂,并且我们需要创建多个这样类的实例时,如果我们用new操作符去创建这样的类实例,这未免会增加创建类的复杂度和耗费更多的内存空间,因为这样在 ...

  2. 设计模式学习系列6 原型模式(prototype)

    原型模式(prototype)用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象.允许一个对象再创建另外一个新对象的时候根本无需知道任何创建细节,只需要请求圆形对象的copy函数皆可. 1 ...

  3. 设计模式(四)原型模式Prototype(创建型)

      设计模式(四)原型模式Prototype(创建型) 1.   概述 我们都知道,创建型模式一般是用来创建一个新的对象,然后我们使用这个对象完成一些对象的操作,我们通过原型模式可以快速的创建一个对象 ...

  4. Java设计模式(4)原型模式(Prototype模式)

    Prototype模式定义:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. Prototype模式允许一个对象再创建另外一个可定制的对象,根本无需知道任何如何创建的细节,工作原理是: ...

  5. 设计模式 --> (6)原型模式

    原型(Prototype)模式 用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象. 原型模式是一种创建型设计模式,Prototype模式允许一个对象再创建另外一个可定制的对象,根本无需知 ...

  6. 设计模式(5)原型模式(Prototype)

    设计模式(0)简单工厂模式 设计模式(1)单例模式(Singleton) 设计模式(2)工厂方法模式(Factory Method) 设计模式(3)抽象工厂模式(Abstract Factory) 设 ...

  7. 面向对象设计模式纵横谈:Prototype 原型模式(笔记记录)

       有一段时间没写东西了,今天继续把没写完的设计模式写完,今天这堂课是创建型设计模式的最后一堂课,原型设计模式,它同样也是解决了对象在创建的过程中的解耦合的情况,面对变化使代码更稳定,更准确的说是使 ...

  8. 设计模式05: Prototype 原型模式(创建型模式)

    Prototype 原型模式(创建型模式) 依赖关系的倒置抽象不应该依赖于实现细节,细节应该依赖于抽象.对所有的设计模式都是这样的. -抽象A直接依赖于实现细节b -抽象A依赖于抽象B,实现细节b依赖 ...

  9. 大型Java进阶专题(五) 设计模式之单例模式与原型模式

    前言 ​ 今天开始我们专题的第四课了,最近公司项目忙,没时间写,今天抽空继续.上篇文章对工厂模式进行了详细的讲解,想必大家对设计模式合理运用的好处深有感触.本章节将介绍:单例模式与原型模式.本章节参考 ...

  10. 【设计模式】Java设计模式精讲之原型模式

    简单记录 - 慕课网 Java设计模式精讲 Debug方式+内存分析 & 设计模式之禅-秦小波 文章目录 1.原型模式的定义 原型-定义 原型-类型 2.原型模式的实现 原型模式的通用类图 原 ...

随机推荐

  1. HTTP Cookie/Session

    一.HTTP协议 HTTP协议是一个无状态协议,服务器无法判断若干个请求是否来自同一个浏览器,无法与浏览器进行会话. 二.HTTP会话控制:Cookie Cookie技术是使用在浏览器端的一种缓存技术 ...

  2. Use the PDFs below or the HTML contents to the left to install and configure P6 EPPM and its additional components.

    Welcome to Your Documentation   Use the PDFs below or the HTML contents to the left to install and c ...

  3. 设计模式--适配器(Adapter)模式

    今天学习另一个设计模式,适配器(Adapter)模式,这是一个共同方向,但有特殊要求,就应用到此设计模式.写到这里,想起很久以前,有写过一篇<ASP.NET的适配器设计模式(Adapter)&g ...

  4. JavaScript异常捕获

    理论准备 ★   异常捕获 △ 异常:当JavaScript引擎执行JavaScript代码时,发生了错误,导致程序停止运行: △ 异常抛出:当异常产生,并且这个异常生成一个错误信息: △ 异常捕获: ...

  5. 三种对话框的示例(alert,confirm,prompt)

    示例代码 <h2>JavaScriptDialog</h2> <hr/> <buttononclick="btn_alert()"> ...

  6. 【C#】线程之Parallel

    在一些常见的编程情形中,使用任务也许能提升性能.为了简化变成,静态类System.Threading.Tasks.Parallel封装了这些常见的情形,它内部使用Task对象. Parallel.Fo ...

  7. Try out the latest C++ compiler toolset without waiting for the next update of Visual Studio

    Updated 22/Apr/2016: The NuGet package is now being uploaded daily. The payload doesn’t change every ...

  8. C#引用C++代码

    现在在Windows下的应用程序开发,VS.Net占据了绝大多数的份额.因此很多以前搞VC++开发的人都转向用更强大的VS.Net.在这种情况下,有很多开发人员就面临了如何在C#中使用C++开发好的类 ...

  9. Python入门笔记(15):对文件的操作(1)

    一.文件对象 我理解的文件对象就是一个接口,通过这个接口对文件进行相关操作. <Python 核心编程>上说的很晦涩,这里没有深刻理解到,希望有人能解释给我听. >>> ...

  10. 004_URL 路由 - URL 路由

    在Web Form 情况下,每一个 ASPX页面既是一个文件,又是一个队请求自包含的响应.而在 MVC 情况下,请求是由控制器类中的动作方法处理的,而且与硬盘上的文件没有一对一的相互关系. ASP.N ...