C++设计模式笔记(08) - Factory Method工厂方法
时间:2022-07-22
本文章向大家介绍C++设计模式笔记(08) - Factory Method工厂方法,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
参考书籍:
《设计模式:可复用面向对象软件的基础》
《Head First 设计模式》
参考课程:《C++设计模式》-李建忠
1.动机(Motivation)
▷在软件系统中,经常面临着创建对象的工作;由于需求的变化,需要创建的对象的具体类型经常变化。
▷如何应对这种变化?如何绕过常规的对象创建方法(new),提供一种“封装机制”来避免客户程序和这种“具体对象创建工作”的紧耦合?
2.模式定义
定义一个用于创建对象的接口,让子类决定实例化哪一个类。Factory Method使得一个类的实例化延迟(目的:解耦,手段:虚函数)到子类。
——《设计模式:可复用面向对象软件的基础》
3.结构(Structure)
4.要点总结
- 所有的工厂都是用来封装对象的创建
- 简单工厂,虽然不是真正的设计模式,但仍不失为一个简单的方法,可以将客户程序从具体类解耦。
- 工厂方法使用继承:把对象的创建委托给子类,子类实现工厂方法来创建对象。
- 抽象工厂使用对象组合:对象的创建被实现在工厂接口所暴露出来的方法中。
- 所有工厂模式都通过减少应用程序和具体类之间的依赖促进松耦合。
- 工厂方法允许类将实例化延迟到子类进行。
- 抽象工厂创建相关的对象家族,而不需要依赖它们的具体类。
- 依赖倒置原则,指导我们避免依赖具体类型,而要尽量依赖抽象。
- 工厂是很有威力的技巧,帮助我们针对抽象编程,而不要针对具体类编程。
5.《Head First 设计模式》实例的C++实现
场景:参照书籍-在建立披萨店之后,我们打算成立分店,建立不同风味的披萨店!
- 抽象工厂类
//PizzaStore.h#ifndef FACTORY_PIZZASTORE_H#define FACTORY_PIZZASTORE_H
#pragma once#include <string>#include "Pizza.h"
using namespace std;
//抽象工厂类class PizzaStore{public: PizzaStore(); ~PizzaStore();
virtual Pizza *orderPizza(string type);
//在PizzaStore里,"工厂方法"现在是抽象的。 virtual Pizza *createPizza(string type) = 0; //现在把工厂对象移到这个方法中。};
#endif //FACTORY_PIZZASTORE_H
//PizzaStore.cpp#include "PizzaStore.h"#include <utility>
PizzaStore::PizzaStore()= default;
PizzaStore::~PizzaStore()= default;
Pizza* PizzaStore::orderPizza(string type){ Pizza* pizza = createPizza(std::move(type)); //现在createPizza()方法从工厂对象中移回PizzaStore
pizza->prepare(); pizza->bake(); pizza->cut(); pizza->box();
return pizza;}
- 抽象产品类
//Pizza.h#ifndef FACTORY_PIZZA_H#define FACTORY_PIZZA_H
#pragma once#include <string>#include <iostream>#include <vector>
using namespace std;
//抽象产品类class Pizza{protected: //每个披萨都具有名称(name)面团类型(dough)酱料类型(sauce)一套佐料(toppings) string name; string dough; string sauce; vector<string> toppings;public: Pizza(); ~Pizza();
virtual void prepare(); virtual void bake(); virtual void cut(); virtual void box(); virtual string& getName();};
#endif //FACTORY_PIZZA_H
//Pizza.cpp#include "Pizza.h"
Pizza::Pizza()= default;
Pizza::~Pizza()= default;
void Pizza::prepare(){ //此抽象类提供了某些默认的基本做法,用来进行烘焙切片装盒 cout << "Preparing " << name << endl; cout << "Tossing dough..." << endl; cout << "Adding sauce..." << endl; cout << "Adding toppings: " << endl;
//准备工作需要以特定的顺序进行,有一连串的步骤 for(const auto & topping : toppings) { cout << " " << topping << endl; }}
//烘焙void Pizza::bake(){ cout << "Bake for 25 minutes at 350" << endl;}
//切片void Pizza::cut(){ cout << "Cutting the pizza into diagonal slices" << endl;}
//装盒void Pizza::box(){ cout << "Place pizza in official PizzaStore box" << endl;}
string& Pizza::getName(){ return name;}
- 具体工厂类1
//NYPizzaStore.h#ifndef FACTORY_NYPIZZASTORE_H#define FACTORY_NYPIZZASTORE_H
#pragma once#include "PizzaStore.h"#include "NYStyleCheesePizza.h"
//具体工厂类//NYPizzaStore扩展PizzaStore,所以拥有odrerPizza()方法(以及其他的方法);class NYPizzaStore : public PizzaStore{public: NYPizzaStore(); ~NYPizzaStore();
//createPizza()返回一个Pizza对象,由子类全权负责该实例化哪一个具体Pizza Pizza* createPizza(string item) override;};#endif //FACTORY_NYPIZZASTORE_H
//NYPizzaStore.cpp#include "NYPizzaStore.h"
NYPizzaStore::NYPizzaStore()= default;
NYPizzaStore::~NYPizzaStore()= default;
Pizza* NYPizzaStore::createPizza(string item){ if(item == "cheese") return new NYStyleCheesePizza;}
- 具体产品类1
//NYStyleCheesePizza.h#ifndef FACTORY_NYSTYLECHEESEPIZZA_H#define FACTORY_NYSTYLECHEESEPIZZA_H
#pragma once#include "pizza.h"
//具体产品类1class NYStyleCheesePizza : public Pizza{public: NYStyleCheesePizza(); ~NYStyleCheesePizza();};
#endif //FACTORY_NYSTYLECHEESEPIZZA_H
//NYStyleCheesePizza.cpp#include "NYStyleCheesePizza.h"
NYStyleCheesePizza::NYStyleCheesePizza(){ //纽约披萨有自己的大葱番茄(Marinara)和薄饼。 name = "NY Style Sauce and Cheese Pizza"; dough = "Thin Crust Dough"; sauce = "Marinara Sauce";
//上面覆盖的是意大利Reggiano高级奶酪 toppings.emplace_back("Grated Reggiano Cheese");}
NYStyleCheesePizza::~NYStyleCheesePizza()= default;
- 具体工厂类2
//ChicagoPizzaStore.h#ifndef FACTORY_CHICAGOPIZZASTORE_H#define FACTORY_CHICAGOPIZZASTORE_H
#pragma once#include "PizzaStore.h"#include "ChicagoStyleCheesePizza.h"
class ChicagoPizzaStore : public PizzaStore{public: ChicagoPizzaStore(); ~ChicagoPizzaStore();
Pizza* createPizza(string type) override;};
#endif //FACTORY_CHICAGOPIZZASTORE_H
//ChicagoPizzaStore.cpp#include "ChicagoPizzaStore.h"
ChicagoPizzaStore::ChicagoPizzaStore()= default;
ChicagoPizzaStore::~ChicagoPizzaStore()= default;
Pizza* ChicagoPizzaStore::createPizza(std::string type){ if (type == "cheese") return new ChicagoStyleCheesePizza; return nullptr;}
- 具体产品类2
//ChicagoStyleCheesePizza.h#ifndef FACTORY_CHICAGOSTYLECHEESEPIZZA_H#define FACTORY_CHICAGOSTYLECHEESEPIZZA_H
#pragma once#include "Pizza.h"
//具体产品类2class ChicagoStyleCheesePizza : public Pizza{public: ChicagoStyleCheesePizza(); ~ChicagoStyleCheesePizza();
void cut() override; //覆盖了cut方法};
#endif //FACTORY_CHICAGOSTYLECHEESEPIZZA_H
//ChicagoStyleCheesePizza.cpp#include "ChicagoStyleCheesePizza.h"
ChicagoStyleCheesePizza::ChicagoStyleCheesePizza(){ //芝加哥披萨使用小番茄作为酱料,并使用厚饼。 name = "Chicago Style Deep Fish Cheese Pizza"; dough = "Extra Thick Crust Dough"; sauce = "Plum Tomato Sauce";
//芝加哥风味的 深盘披萨使用许多意大利白干酪(mozzarella) toppings.emplace_back("Shredded Mozzarella Cheese");}
ChicagoStyleCheesePizza::~ChicagoStyleCheesePizza()= default;
void ChicagoStyleCheesePizza::cut(){ //这个芝加哥风味披萨覆盖了cut()方法,将披萨切成正方形 cout << "Cutting the pizza into square slices" << endl;}
- 测试
//main.cpp//工厂方法模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。//工厂方法让类把实例化推迟到子类。//设计原则:要依赖抽象,不要依赖具体类#include <iostream>#include "NYPizzaStore.h"#include "ChicagoPizzaStore.h"
using namespace std;
//测试int main(){ //建立两个不同的店 NYPizzaStore nyStore; ChicagoPizzaStore chicagoStore;
Pizza *pizza = nyStore.orderPizza("cheese"); cout << "Ethan ordered a " << pizza->getName() << "n" << endl; delete pizza;
pizza = chicagoStore.orderPizza("cheese"); cout << "Joel ordered a " << pizza->getName() << "n" << endl; delete pizza;
return 0;}
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- 带命令行控制台的轻量级日志分析工具-GoAccess
- 七日Python之路--第八天
- Go 语言学习之运算符
- 神奇!如何快速成为一名优秀的YAML工程师?
- 移动端H5开发入门
- Go 语言学习之流程控制
- MinGW 安装
- 实践 | 目前最快精度最高检测框架(EfficientDet)
- 解决拉取github仓库报错“gnutls_handshake() failed”问题
- HTML 学习
- Go 语言学习之数组
- 坐姿不对,屏幕就变模糊!小姐姐教你用TensorFlow做一款“隐形背背佳”
- 七日Python之路--第九天(blog与Django)
- Python数据可视化-seaborn Iris鸢尾花数据
- pythonGUI -- pyside安装与初试