当前位置: 首页 > news >正文

[设计模式] 4 原型模式 prototype

 

设计模式:可复用面向对象软件的基础》(DP)本文介绍原型模式模板方法模式的实现。首先介绍原型模式,然后引出模板方法模式。

       DP书上的定义为:用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。其中有一个词很重要,那就是拷贝。可以说,拷贝是原型模式的精髓所在。举个现实中的例子来介绍原型模式。找工作的时候,我们需要准备简历。假设没有打印设备,因此需手写简历,这些简历的内容都是一样的。这样有个缺陷,如果要修改简历中的某项,那么所有已写好的简历都要修改,工作量很大。随着科技的进步,出现了打印设备。我们只需手写一份,然后利用打印设备复印多份即可。如果要修改简历中的某项,那么修改原始的版本就可以了,然后再复印。原始的那份手写稿相当于是一个原型,有了它,就可以通过复印(拷贝)创造出更多的新简历。这就是原型模式的基本思想。下面给出原型模式的UML图,以刚才那个例子为实例。

        原型模式实现的关键就是实现Clone函数,对于C++来说,其实就是拷贝构造函数,需实现深拷贝,下面给出一种实现。

 

 

#include <iostream>
#include <string.h>
using namespace std;

class Resume
{
    protected:
        char *name;
    public:
        Resume() {}
        virtual ~Resume() {}
        virtual Resume* Clone() { return NULL; }
        virtual void Set(char *n) {}
        virtual void Show() {}
};

class ResumeA : public Resume
{
    public:
        ResumeA(const char *str);  //ctor
        ResumeA(const ResumeA &r); //copy cotr
        ~ResumeA();                //dector
        ResumeA* Clone();          //copy, key
        void Show();               //show
};
ResumeA::ResumeA(const char *str)
{
    if(str == NULL) {
        name = new char[1];
        name[0] = '\0';
    }
    else {
        name = new char[strlen(str)+1];
        strcpy(name, str);
    }
}
ResumeA::~ResumeA() { delete [] name;}
ResumeA::ResumeA(const ResumeA &r) {
    name = new char[strlen(r.name)+1];
    strcpy(name, r.name);
}
ResumeA* ResumeA::Clone() {
    return new ResumeA(*this);
}
void ResumeA::Show() {
    cout<<"ResumeA name : "<<name<<endl;
}

class ResumeB : public Resume
{
    public:
        ResumeB(const char *str);  //ctor
        ResumeB(const ResumeB &r); //copy cotr
        ~ResumeB();                //dector
        ResumeB* Clone();          //copy, key
        void Show();               //show
};
ResumeB::ResumeB(const char *str)
{
    if(str == NULL) {
        name = new char[1];
        name[0] = '\0';
    }
    else {
        name = new char[strlen(str)+1];
        strcpy(name, str);
    }
}
ResumeB::~ResumeB() { delete [] name;}
ResumeB::ResumeB(const ResumeB &r) {
    name = new char[strlen(r.name)+1];
    strcpy(name, r.name);
}
ResumeB* ResumeB::Clone() {
    return new ResumeB(*this);
}
void ResumeB::Show() {
    cout<<"ResumeB name : "<<name<<endl;
}

int main()
{
    Resume *r1 = new ResumeA("A");
    Resume *r2 = new ResumeB("B");
    Resume *r3 = r1->Clone();
    Resume *r4 = r2->Clone();
    r1->Show(); r2->Show();
    //delete r1,r2
    delete r1; delete r2;
    r1 = r2 = NULL;
    //deep copy for r3, r4
    r3->Show(); r4->Show();
    delete r3; delete r4;
    r3 = r4 = NULL;
}

 

 

最近有个招聘会,可以带上简历去应聘了。但是,其中有一家公司不接受简历,而是给应聘者发了一张简历表,上面有基本信息、教育背景、工作经历等栏,让应聘者按照要求填写完整。每个人拿到这份表格后,就开始填写。如果用程序实现这个过程,该如何做呢?一种方案就是用模板方法模式:定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。我们的例子中,操作就是填写简历这一过程,我们可以在父类中定义操作的算法骨架,而具体的实现由子类完成。下面给出它的UML图。

       其中FillResume() 定义了操作的骨架,依次调用子类实现的函数。相当于每个人填写简历的实际过程。接着给出相应的C++代码。

#include <iostream>
#include <string.h>
using namespace std;

class Resume
{
    protected:
        virtual void SetPersonalInfo() {}
        virtual void SetEducation() {}
        virtual void SetWorkExp() {}
    public:
        void FillResume() 
        {   
            SetPersonalInfo();
            SetEducation();
            SetWorkExp();
        }   
};
class ResumeA: public Resume
{
    protected:
        void SetPersonalInfo() { cout<<"A's PersonalInfo"<<endl; }
        void SetEducation() { cout<<"A's Education"<<endl; }
        void SetWorkExp() { cout<<"A's Work Experience"<<endl; }
};
class ResumeB: public Resume
{
    protected:
        void SetPersonalInfo() { cout<<"B's PersonalInfo"<<endl; }
        void SetEducation() { cout<<"B's Education"<<endl; }
        void SetWorkExp() { cout<<"B's Work Experience"<<endl; }
};
int main()
{
    Resume *r1;
    r1 = new ResumeA();
    r1->FillResume();
    delete r1;
    r1 = new ResumeB();
    r1->FillResume();
    delete r1;
    r1 = NULL;
    return 0;
}

 

Prototype 模式通过复制原型(Prototype)而获得新对象创建的功能,这里 Prototype 本
身就是“对象工厂”(因为能够生产对象),实际上 Prototype 模式和 Builder 模式、
AbstractFactory 模式都是通过一个类(对象实例)来专门负责对象的创建工作(工厂对象),
它们之间的区别是:Builder 模式重在复杂对象的一步步创建(并不直接返回对象),
AbstractFactory 模式重在产生多个相互依赖类的对象,而 Prototype 模式重在从自身复制自
己创建新类。

相关文章:

  • C++ 二叉树遍历实现
  • 猴子排序 然而并没有什么用 233
  • Java内存管理和垃圾回收
  • HTML5 实现Link跳线效果
  • 计算机网络的现实应用
  • .NET “底层”异步编程模式——异步编程模型(Asynchronous Programming Model,APM)...
  • monkey如何通过uiautomatorviewer的bounds坐标点击控件
  • 正则表达式 java版
  • Linux管理命令
  • Windows下Composer使用问题
  • excel表格生成与导入
  • 【IOS】封装了AFNetworking
  • Linux驱动开发 -- 打开dev_dbg()
  • 处理hadoop发送的文件到后台并解析存储到数据库策略
  • 猫猫学IOS(三十)UI之Quartz2D画图片画文字
  • Android框架之Volley
  • Centos6.8 使用rpm安装mysql5.7
  • fetch 从初识到应用
  • iOS 系统授权开发
  • iOS仿今日头条、壁纸应用、筛选分类、三方微博、颜色填充等源码
  • Node.js 新计划:使用 V8 snapshot 将启动速度提升 8 倍
  • python_bomb----数据类型总结
  • Spring核心 Bean的高级装配
  • 关于Flux,Vuex,Redux的思考
  • 关于List、List?、ListObject的区别
  • 名企6年Java程序员的工作总结,写给在迷茫中的你!
  • 前端设计模式
  • 浅谈Golang中select的用法
  • 微信开放平台全网发布【失败】的几点排查方法
  • 微信小程序--------语音识别(前端自己也能玩)
  • 一份游戏开发学习路线
  • 一加3T解锁OEM、刷入TWRP、第三方ROM以及ROOT
  • 运行时添加log4j2的appender
  • 在GitHub多个账号上使用不同的SSH的配置方法
  • 在Unity中实现一个简单的消息管理器
  • 主流的CSS水平和垂直居中技术大全
  • kubernetes资源对象--ingress
  • 回归生活:清理微信公众号
  • 微龛半导体获数千万Pre-A轮融资,投资方为国中创投 ...
  • (1)(1.11) SiK Radio v2(一)
  • (C++17) optional的使用
  • (LeetCode C++)盛最多水的容器
  • (初研) Sentence-embedding fine-tune notebook
  • (十一)手动添加用户和文件的特殊权限
  • (四)模仿学习-完成后台管理页面查询
  • (循环依赖问题)学习spring的第九天
  • (转) ns2/nam与nam实现相关的文件
  • (转)程序员疫苗:代码注入
  • ***微信公众号支付+微信H5支付+微信扫码支付+小程序支付+APP微信支付解决方案总结...
  • *++p:p先自+,然后*p,最终为3 ++*p:先*p,即arr[0]=1,然后再++,最终为2 *p++:值为arr[0],即1,该语句执行完毕后,p指向arr[1]
  • ... fatal error LINK1120:1个无法解析的外部命令 的解决办法
  • .describe() python_Python-Win32com-Excel
  • .NET NPOI导出Excel详解
  • .NET/C# 中设置当发生某个特定异常时进入断点(不借助 Visual Studio 的纯代码实现)
  • .NET建议使用的大小写命名原则