1. 程式人生 > >C++設計模式 ==> 原型模式

C++設計模式 ==> 原型模式

簡介

        所謂原型模式就是用原型例項指定建立物件的種類,並且通過複製這些原型建立新的物件。舉一個簡單的例子,一個人想要批量分發自己的名片,然而名片千篇一律,只是偶爾修改一些地方,但需求量大,所以需要不斷例項化名片類,十分麻煩,所以這就有了原型模式。原型模式旨在以某一類為原型,利用Clone成員函式不斷拷貝出新物件。
        在C++中,有淺拷貝和深拷貝兩種拷貝模式,預設賦值採用淺拷貝的方法,但淺拷貝會共享分配的堆疊空間,析構時會兩次釋放空間導致程式崩潰,所以原型模式一般直接使用深拷貝,由拷貝建構函式實現。
        下面我們就以印名片為例子為大家講解一下原型模式的用法。

圖示

原型模式圖示

程式碼實現

////////////////////////////
//
// FileName : ProtoTypeDefine.h
// Editor : PeterZheng
// Date : 2018/8/16 18:00
//
////////////////////////////

#pragma once

#ifndef PROTOTYPE_H_
#define PROTOTYPE_H_

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <string>
#include <windows.h>

#endif

#define MAX_LENGTH 255
//克隆抽象類 class ProtoType { public: virtual ProtoType* Clone() = 0; }; //克隆具體實現類(名片類) class ConcreteProtoType { public: int age; char* name; char* school; ConcreteProtoType(int a, const char* name, const char* school) { this->age = a; this->name = new char[MAX_LENGTH]; this
->school = new char[MAX_LENGTH]; ZeroMemory(this->name, MAX_LENGTH); ZeroMemory(this->school, MAX_LENGTH); strcpy_s(this->name, MAX_LENGTH, name); strcpy_s(this->school, MAX_LENGTH, school); } ConcreteProtoType(const ConcreteProtoType& pt) { this->age = pt.age; this->name = new char[MAX_LENGTH]; this->school = new char[MAX_LENGTH]; ZeroMemory(this->name, MAX_LENGTH); ZeroMemory(this->school, MAX_LENGTH); strcpy_s(this->name, MAX_LENGTH, pt.name); strcpy_s(this->school, MAX_LENGTH, pt.school); } ~ConcreteProtoType() {} ConcreteProtoType* Clone() { return new ConcreteProtoType(*this); } void Show() { std::cout << "Name: " << this->name << std::endl << "Age: " << this->age << std::endl << "School: " << this->school << std::endl; } };
////////////////////////////
//
// FileName : ProtoTypeDemo.cpp
// Editor : PeterZheng
// Date : 2018/8/16 19:22
//
////////////////////////////

#include "ProtoTypeDefine.h"

using namespace std;

int main(void)
{
    ConcreteProtoType *mp1 = new ConcreteProtoType(28, "XiaoMin", "CMU");
    ConcreteProtoType *mp2 = mp1->Clone();
    //名片的學校需要改為MIT
    ConcreteProtoType *mp3 = new ConcreteProtoType(28, "XiaoMin", "MIT");
    ConcreteProtoType *mp4 = mp3->Clone();
    mp1->Show();
    cout << endl;
    mp2->Show();
    cout << endl;
    mp3->Show();
    cout << endl;
    system("pause");
    return 0;
}