设计模式学习笔记九:原型模式(Prototype Pattern)

1.概述
    
意图:我们将已经存在的对象作为原型,用户可以通过复制这些原型创建新的对象。
    
使用场合:当一个系统应该独立于产品的创建、构造和表示时,可以使用原型模式。在原型模式中,产品的创建和初始化再类的Clone方法中完成。在使用是,我们可以用一些列原型对象来代替生成相应对象的工厂对象,并且可以使拷贝、粘贴等操作独立于需要复制的对象。
    
结构:
    
原型模式(Prototype):用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。原型模式说白了就是从一个对象再创建另外一个可定制的对象,而且不需要直到任何创建的细节。
     
    原型模式基本代码:
    原型类:
Code
public abstract class Prototype
    
{
        
private string id;

        
// Constructor 
        public Prototype(string id)
        
{
            
this.id = id;
        }


        
// Property 
        public string Id
        
{
            
get return id; }
        }


        
public abstract Prototype Clone();
    }

    具体原型类:

Code
public class ConcretePrototype1 : Prototype
    
{
        
// Constructor 
        public ConcretePrototype1(string id)
            : 
base(id)
        
{
        }


        
public override Prototype Clone()
        
{
            
// Shallow copy 
            return (Prototype)this.MemberwiseClone();
        }

    }



    
public class ConcretePrototype2 : Prototype
    
{
        
// Constructor 
        public ConcretePrototype2(string id)
            : 
base(id)
        
{
        }


        
public override Prototype Clone()
        
{
            
// Shallow copy 
            return (Prototype)this.MemberwiseClone();
        }

    }

    客户端:
Code
            ConcretePrototype1 p1 = new ConcretePrototype1("I");
            ConcretePrototype1 c1 
= (ConcretePrototype1)p1.Clone();
            Console.WriteLine(
"Cloned: {0}", c1.Id);

            ConcretePrototype2 p2 
= new ConcretePrototype2("II");
            ConcretePrototype2 c2 
= (ConcretePrototype2)p2.Clone();
            Console.WriteLine(
"Cloned: {0}", c2.Id);
    2. 实例
    
对于.NET而言,原型模式抽象类Prototype是用不着的,在.NETSystem命名空间中提供了ICloneable接口,其中就是唯一的一个方法Clone(),这样我们只需要实现这个接口就可以完成原型模式了。
    
下面看大话设计模式中的简历的原型实现:
代码结构图:

    简历类:
Code
public class Resume : ICloneable
    
{
        
private string name;
        
private string sex;
        
private string age;
        
private string timeArea;
        
private string company;

        
public Resume(string name)
        
{
            
this.name = name;
        }


        
//设置个人信息
        public void SetPersonalInfo(string sex, string age)
        
{
            
this.sex = sex;
            
this.age = age;
        }

        
//设置工作经历
        public void SetWorkExperience(string timeArea, string company)
        
{
            
this.timeArea = timeArea;
            
this.company = company;
        }


        
//显示
        public void Display()
        
{
            Console.WriteLine(
"{0} {1} {2}", name, sex, age);
            Console.WriteLine(
"工作经历:{0} {1}", timeArea, company);
        }


        
public Object Clone()
        
{
            
return (Object)this.MemberwiseClone();
        }


}

    客户端调用:

Code
static void Main(string[] args)
{
    Resume a 
= new Resume("大鸟");
    a.SetPersonalInfo(
"""29");
    a.SetWorkExperience(
"1998-2000""XX公司");
    Resume b 
= (Resume)a.Clone();
    b.SetWorkExperience(
"1998-2006""YY企业");
    Resume c 
= (Resume)a.Clone();
    c.SetPersonalInfo(
"""24");
    a.Display();
    b.Display();
    c.Display();
    Console.Read();
}

    结果显示:
    大鸟 29
    工作经历 1998-2000 XX公司
    大鸟 29
    工作经历 1998-2006 YY公司
    大鸟 24
    工作经历 1998-2000 XX公司
    
一般在初始化的信息不发生变化的情况下,克隆是最好的方法。这既隐藏了对象的创建细节,又对性能是大大的提高。
    
下面我们来看深克隆和浅克隆:
    在上面的简历类中,数据都是string型的,而string是一种拥有值类型特点的特殊引用类型,MemberwiseClone()方法对于值类型的字段执行逐位复制,对于引用类型,则只复制引用的对象,因此,原对象及其副本引用同一个对象。我们看下面的引用类型的简历克隆的代码实现:
    
代码结构图:



    详细代码:
    工作经历类:
Code
//工作经历
    public class WorkExperience
    
{
        
private string workDate;
        
public string WorkDate
        
{
            
get return workDate; }
            
set { workDate = value; }
        }

        
private string company;
        
public string Company
        
{
            
get return company; }
            
set { company = value; }
        }

    }


    简历类:
Code
//简历
    public class Resume : ICloneable
    
{
        
private string name;
        
private string sex;
        
private string age;

        
private WorkExperience work;

        
public Resume(string name)
        
{
            
this.name = name;
            work 
= new WorkExperience();
        }


        
//设置个人信息
        public void SetPersonalInfo(string sex, string age)
        
{
            
this.sex = sex;
            
this.age = age;
        }

        
//设置工作经历
        public void SetWorkExperience(string workDate, string company)
        
{
            work.WorkDate 
= workDate;
            work.Company 
= company;
        }


        
//显示
        public void Display()
        
{
            Console.WriteLine(
"{0} {1} {2}", name, sex, age);
            Console.WriteLine(
"工作经历:{0} {1}", work.WorkDate, work.Company);
        }


        
public Object Clone()
        
{
            
return (Object)this.MemberwiseClone();
        }


    }


    客户端:
      static void Main(string[] args)
        
{
            Resume a 
= new Resume("大鸟");
            a.SetPersonalInfo(
"""29");
            a.SetWorkExperience(
"1998-2000""XX公司");

            Resume b 
= (Resume)a.Clone();
            b.SetWorkExperience(
"1998-2006""YY企业");

            Resume c 
= (Resume)a.Clone();
            c.SetPersonalInfo(
"""24");
            c.SetWorkExperience(
"1998-2003""ZZ企业");

            a.Display();
            b.Display();
            c.Display();

            Console.Read();
        }

    

下面我们看运行结果: 
    大鸟 29
    工作经历 1998-2003 ZZ企业
    
大鸟 29
     
工作经历 1998-2003 ZZ企业
    
大鸟 24
     
工作经历 1998-2003 ZZ企业

    由于MemberwiseClone()方法是浅表复制(克隆),对于值类型克隆没有问题,对于引用类型对象,只复制了引用,对引用的对象还是指向了原来的对象,所以就会出现我给abc三个引用设置‘工作经历’,但却同时看到三个引用都是最后一次设置,因为三个引用都指向了同一个对象。
    
“浅复制”,被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用都仍然指向原来的对象。
    
“深复制”,深复制把引用对象的变量指向复制过的对象,而不是原有的被引用的对象。
    
下面来看深复制的实现:
    
代码结构图:


    实现代码:
    
工作经验类:

   //工作经历
    public class WorkExperience : ICloneable
    
{
        
private string workDate;
        
public string WorkDate
        
{
            
get return workDate; }
            
set { workDate = value; }
        }

        
private string company;
        
public string Company
        
{
            
get return company; }
            
set { company = value; }
        }


        
public Object Clone()
        
{
            
return (Object)this.MemberwiseClone();
        }

    }

    简历类:

Code
 //简历
    public class Resume : ICloneable
    
{
        
private string name;
        
private string sex;
        
private string age;

        
private WorkExperience work;

        
public Resume(string name)
        
{
            
this.name = name;
            work 
= new WorkExperience();
        }


        
private Resume(WorkExperience work)
        
{
            
this.work = (WorkExperience)work.Clone();
        }


        
//设置个人信息
        public void SetPersonalInfo(string sex, string age)
        
{
            
this.sex = sex;
            
this.age = age;
        }

        
//设置工作经历
        public void SetWorkExperience(string workDate, string company)
        
{
            work.WorkDate 
= workDate;
            work.Company 
= company;
        }


        
//显示
        public void Display()
        
{
            Console.WriteLine(
"{0} {1} {2}", name, sex, age);
            Console.WriteLine(
"工作经历:{0} {1}", work.WorkDate, work.Company);
        }


        
public Object Clone()
        
{
            Resume obj 
= new Resume(this.work);

            obj.name 
= this.name;
            obj.sex 
= this.sex;
            obj.age 
= this.age;


            
return obj;
        }


    }

    客户端代码与上面相同,执行结果:
    
大鸟 29
    
工作经历 1998-2000 XX公司
    大鸟 29
    
工作经历 1998-2006 YY企业
    
大鸟 24
    
工作经历 1998-2003 ZZ企业
    3. 总结
    
优缺点:
    
使用原型模式有以下优点:
    
1)。在运行时增加或删除产品,只要通过客户端注册原型实例即可将新产品类型增加到系统中,例如组态软件中工具箱中的每个工具可以对应一个注册的原型对象,可以通过增加原型对象扩展工具箱。
    
2)。很容易的创建复杂的对象:在图像编辑和组态等软件中,经常需要创建复杂的图元,这些图元是由简单的图元组成的,采用原型模式可以很容易的将复杂图元作为一般图元来使用,是软件的工具箱具有扩展功能。
    
3)。减少工厂的层次:由于在.NET中可以使用反射工厂,因此这个优势并不明显。
    
使用原型模式的缺点:是在有些情况下克隆功能不容易实现,特别是在遇到对象的循环引用时。
    
.NET中的很多类支持原型模式,例如我们希望获得一个与现有数据集(DataSet)结构相同的数据集,既可以采用克隆的方法。注意,DataSetClone()Copy()两个方法,Clone()方法用来复制DataSet的结构,但不复制DataSet的数据,实现了原型模式的浅复制,Copy()方法,不但复制结构,也复制数据,实现了原型模式的深复制。

转载于:https://www.cnblogs.com/peida/archive/2008/06/26/1230129.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/276865.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Centos7上安装docker

步骤:1、Docker 要求 CentOS 系统的内核版本高于 3.10 ,查看本页面的前提条件来验证你的CentOS 版本是否支持 Docker 。通过 uname -r 命令查看你当前的内核版本2、使用 root 权限登录 Centos。确保 yum 包更新到最新。 (这个可能需要几分钟的…

pythonista3安装stash_Pythonista下stash安装教程

前言 “StaSh is a serious attempt to implement a Bash-like shell for Pythonista.” StaSh是一个Pythonista环境下的仿shell程序,Sta来自于Pythonista的后三个字母,Sh即shell缩写。除了能完成shell的基本功能外,最主要的功能还有实现pip安…

c++ map用法_Python的 5 种高级用法,效率提升没毛病

原创:机器之心(ID:almosthuman2014)任何编程语言的高级特征通常都是通过大量的使用经验才发现的。比如你在编写一个复杂的项目,并在 stackoverflow 上寻找某个问题的答案。然后你突然发现了一个非常优雅的解决方案,它使用了你从不…

非对称加密算法RSA加密传输数据python3源代码实现

2019独角兽企业重金招聘Python工程师标准>>> import rsa# RSA 算法规定: # 待加密的字节数不能超过密钥的长度值除以 8 再减去 11NBIT 4096 CAN_ENCODE_LEN NBIT // 8 - 11 PER_ENCODE_LEN CAN_ENCODE_LEN - (CAN_ENCODE_LEN % 2) PER_DECODE_LEN CA…

(Microsoft) Visual Studio LightSwitch

在蓝色小铺,听到了前辈 阿源哥哥提到 Visual Studio LightSwitch"号称" 可以快速开发桌面、云端的应用程序。http://www.microsoft.com/visualstudio/en-us/lightswitch (这里也提供下载) 原厂提供的图片: 跟「正…

jsp调用controller方法_RPC调用_服务注册与发现

RPC调用_单体架构_SOA架构系统架构的演变 1 传统的单体架构 1.1 什么是单体架构 一个归档包(例如 war 格式或者 Jar 格式)包含了应用所有功能的应用程序,我们通常称之 为单体应用。也称之为单体应用架构,这是一种比较传统的架构风…

MFC项目实战(1)文件管理器--准备篇

本程序主要实现如下功能: 程序通过左边的树形控件显示本地计算机中目录的结构,右边的列表控件则负责响应树形控件中选择的目录节点并把此节点中的所有项在列表框中显示出来,列表框支持奇偶行颜色设置,选中颜色设置和热点颜色设置&…

使用tab键分割的文章能快速转换成表格。( )_EXCEL的163种使用技巧集锦-42~62

本文主要讲述了EXCEL的163种使用技巧其中20条技巧,待163种使用技巧全部讲述完成后,如有需要,各位看官可在评论中留下邮箱,小编会将整理的WORD版发给大家。42. 快速查找工作簿 你可以利用在工作表中的任何文字进行搜寻,…

android通过代码设置铃声_第六十四回:Android中UI控件之SeekBar

各位看官们,大家好,上一回中咱们说的是Android中UI控件之ProgressBar的例子,这一回咱们的例子是UI控件之SeekBar。闲话休提,言归正转。让我们一起Talk Android吧!看官们,SeekBar通常翻译为拖动条&#xff0…

利旧IBM X3650m2 安装esxi5.1提供云桌面(备忘)

上面左边是我的个人微信,如需进一步沟通,请加微信。 右边是我的公众号“Openstack私有云”,如有兴趣,请关注。公司有一台旧服务器,原来的IBM X3650m2 服务器,2物理CPU,24G内存,8块3…

asp登录页面跳转到注册页面_Java 添加页面跳转按钮到PDF文档

概述当我们在查阅含有大量页面的PDF时,可通过在页面上添加跳转按钮来实现页面转换,以达到节约时间,提高效率的目的。本文将通过Java程序来演示如何给PDF文档添加页面跳转按钮。通常来说跳转可分为两种情况:一是跳转至特殊页面(首页…

确定最佳聚类数matlab代码_详解DBSCAN聚类

使用DBSCAN标识为员工分组照片由Ishan seefromthesky 在 Unsplash拍摄基于密度的噪声应用空间聚类(DBSCAN)是一种无监督的ML聚类算法。无监督的意思是它不使用预先标记的目标来聚类数据点。聚类是指试图将相似的数据点分组到人工确定的组或簇中。它可以替代KMeans和层次聚类等流…

Arrays.asList 使用细节

通常初始化后使用如下,但是报错 UnsupportOperationException.... 根据提示信息,就是调用add()方法时抛出了异常。顺着堆栈信息往上找,提示的是AbstractList类的108行出了异常,这一行所在方法的具体实现如下: //108行 …

python list元素合并_python list 合并连接字符串的方法

python list 合并连接字符串的方法 更新时间:2013年03月09日 22:02:18 作者: python 列表合并字符串,我们一般会用到字符串的join方法来操作。下面通过代码的形式,详细的说下list怎么拼成字符串? 相关文章这篇文章主要…

单片机按键防抖程序_这些单片机按键设计方案,请拿好,不谢!

在单片机系统里,按键是常见的输入设备,在本文中介绍几种按键硬件、软件设计方面的技巧。一般的在按键的设计上,一般有四种方案,创客学院带你零基础学习电子产品设计。一是GPIO口直接检测单个按键,如图1.1所示;二是按键…

Oracle 11G 安装详解

oracle官网下载地址:http://www.oracle.com/technetwork/database/enterprise-edition/downloads/index.html 官网下载需要注册orcale账号,比较繁琐,这里直接放已经下载好的安装包链接 Oracle11G下载链接: https://pan.baidu.com/s/1v6oD4jAt…

kubeadm部署k8s_用 kubeadm 部署生产级 k8s 集群

概述kubeadm 已⽀持集群部署,且在1.13 版本中 GA,⽀持多 master,多 etcd 集群化部署,它也是官⽅最为推荐的部署⽅式,⼀来是由它的 sig 组来推进的,⼆来 kubeadm 在很多⽅⾯确实很好的利⽤了 kubernetes 的许…

统计学习方法概论---分类问题

为什么80%的码农都做不了架构师?>>> 分类问题 转载于:https://my.oschina.net/liyangke/blog/2945185

CENTOS7 Python3.7 PyAudio 安装

2019独角兽企业重金招聘Python工程师标准>>> 出现错误: gcc -pthread -Wno-unused-result -Wsign-compare -DNDEBUG -g -fwrapv -O3 -Wall -fPIC -fPIC -I/usr/local/python371/include/python3.7m -c src/_portaudiomodule.c -o build/temp.linux-aarch64-3.7/src/…