经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » Java相关 » 设计模式 » 查看文章
深入浅出《设计模式》之工厂模式(C++)
来源:cnblogs  作者:尔广强  时间:2019/7/19 9:01:02  对本文有异议

前言

模式介绍

在之前简单工厂模式中,我们介绍了简单工厂模式的缺陷是违背了开放-封闭原则。如果在面馆中添加了烤海参,那将会修改waiter工厂类。违背了类内封闭原则。

还以面馆为例,现在两种面,用一个服务员来卖就可以,如果这个服务员不干了,后面卖面的厨师需要兼职顶替服务员,但是厨师又不能离开灶台,就将模式改成了窗口排队式,一队海参炒面,另一队辣根汤面。每个窗口分别有一个厨师放饭,一个会做海参炒面,另一个会做辣根汤面。老板觉得这种模式挺好,万一来了一个会做烤海参的,就只需要开一个烤海参的窗口就好了,不需要重新要服务员学习,因为烤海参的就会卖。这就变成了工厂模式。

UML类图

这里涉及到2种类①我,客户端,排队买饭。②做饭厨师类,工厂类,为我生成饭。③菜品类,生成菜品类。具体关系如下UML类图:

代码实例

下面是noodle类,是为工厂类使用的,继承他就可以扩展noodle类别:

  1. #ifndef NOODLE_H
  2. #define NOODLE_H
  3. class noodle
  4. {
  5. public:
  6. noodle() {}
  7. ~noodle() {}
  8. virtual void eating() = 0;
  9. };
  10. #endif // NOODLE_H

下面是海参炒面类,继承了noodle,实现eating方法,吃海参炒面:

  1. #ifndef HAISHENNOODLE_H
  2. #define HAISHENNOODLE_H
  3. #include "noodle.h"
  4. class haishennoodle : public noodle
  5. {
  6. public:
  7. haishennoodle();
  8. ~haishennoodle();
  9. virtual void eating();
  10. };
  11. #endif // HAISHENNOODLE_H
  1. #include <iostream>
  2. #include "haishennoodle.h"
  3. haishennoodle::haishennoodle()
  4. {
  5. }
  6. haishennoodle::~haishennoodle()
  7. {
  8. }
  9. void haishennoodle::eating()
  10. {
  11. std::cout << "我是海参炒面,里面没有海参哦!!吃的时候注意!" << std::endl;
  12. }

下面是辣根汤面,继承了noodle,实现eating方法,吃辣根汤面:

  1. #ifndef LAGENNOODLE_H
  2. #define LAGENNOODLE_H
  3. #include "noodle.h"
  4. class lagennoodle : public noodle
  5. {
  6. public:
  7. lagennoodle();
  8. ~lagennoodle();
  9. virtual void eating();
  10. };
  11. #endif // LAGENNOODLE_H
  1. #include <iostream>
  2. #include "lagennoodle.h"
  3. lagennoodle::lagennoodle()
  4. {
  5. }
  6. lagennoodle::~lagennoodle()
  7. {
  8. }
  9. void lagennoodle::eating()
  10. {
  11. std::cout << "我是辣根汤面,吃完呛的哼啊!!!" << std::endl;
  12. }

下面是waiter工厂的基类。所有工厂都继承这个类:

  1. #ifndef WAITER_H
  2. #define WAITER_H
  3. class noodle;
  4. class waiter
  5. {
  6. public:
  7. waiter() {}
  8. ~waiter() {}
  9. virtual noodle *createnoodle() = 0;
  10. };
  11. #endif // WAITER_H

下面是海参厨师(工厂1),海参厨师只管做海参炒面,重写了createnoodle方法:

  1. #ifndef HAISHEN_H
  2. #define HAISHEN_H
  3. #include "waiter.h"
  4. class noodle;
  5. class haishen : public waiter
  6. {
  7. public:
  8. haishen();
  9. ~haishen();
  10. virtual noodle *createnoodle();
  11. };
  12. #endif // HAISHEN_H
  1. #include <iostream>
  2. #include "haishen.h"
  3. #include "haishennoodle.h"
  4. haishen::haishen()
  5. {
  6. }
  7. haishen::~haishen()
  8. {
  9. }
  10. noodle *haishen::createnoodle()
  11. {
  12. std::cout << "面是我炒得,我的名字叫海参!!!" << std::endl;
  13. return new haishennoodle();
  14. }

下面是辣根厨师(工厂1),辣根厨师只管做辣根汤面,重写了createnoodle方法:

  1. #ifndef LAGEN_H
  2. #define LAGEN_H
  3. #include "waiter.h"
  4. class lagen : public waiter
  5. {
  6. public:
  7. lagen();
  8. ~lagen();
  9. virtual noodle *createnoodle();
  10. };
  11. #endif // LAGEN_H
  1. #include <iostream>
  2. #include "lagen.h"
  3. #include "lagennoodle.h"
  4. lagen::lagen()
  5. {
  6. }
  7. lagen::~lagen()
  8. {
  9. }
  10. noodle *lagen::createnoodle()
  11. {
  12. std::cout << "吃辣根汤面,你不觉得呛得哼吗??" << std::endl;
  13. return new lagennoodle();
  14. }

下面是客户端,客户端通过类别,使用相应的工厂类建立相应的实例:

  1. #include <iostream>
  2. #include <string.h>
  3. #include "haishen.h"
  4. #include "lagen.h"
  5. #include "noodle.h"
  6. using namespace std;
  7. char *product_list[] = {
  8. "haishen-noodle",
  9. "lagen-noodle",
  10. NULL
  11. };
  12. int main()
  13. {
  14. char *p = NULL;
  15. char *pd = "haishen-noodle";
  16. int i = 0;
  17. waiter *w = NULL;
  18. noodle *n = NULL;
  19. for(p = product_list[i]; p != NULL; i++, p = product_list[i]) {
  20. if(strncmp(pd, p, strlen(pd)) == 0) {
  21. if(i == 0) {
  22. w = new haishen();
  23. } else if(i == 1) {
  24. w = new lagen();
  25. } else {
  26. cout << "对不起,请您排在队内!!!" << std::endl;
  27. break;
  28. }
  29. }
  30. }
  31. if(w) n = w->createnoodle();
  32. if(n) n->eating();
  33. if(w) {
  34. delete w; w = NULL;
  35. }
  36. if(n) {
  37. delete n; n = NULL;
  38. }
  39. return 0;
  40. }

下面是CMakeList.txt文件,帮助生成Makefile:

  1. cmake_minimum_required(VERSION 2.8)
  2. project(noodle-factory)
  3. set(SRC_LIST main.cpp noodle.h waiter.h haishen.h haishen.cpp haishennoodle.h haishennoodle.cpp
  4. lagennoodle.h lagennoodle.cpp lagen.h lagen.cpp)
  5. add_executable(${PROJECT_NAME} ${SRC_LIST})

编译运行结果

代码下载链接是:https://github.com/erguangqiang/freesir_headfirst/blob/master/noodle-factory.tar.gz
使用cmake生成Makefile,并编译出可执行程序noodle。运行结果如下:

  1. erguangqiang@elab$./noodle-factory
  2. 面是我炒得,我的名字叫海参!!!
  3. 我是海参炒面,里面没有海参哦!!吃的时候注意!

结束

工厂模式解决了简单工厂违背了的开放-封闭原则。虽然累的结构变的复杂了,但是对于扩展性得到了很大的提高。

原文链接:http://www.cnblogs.com/freeman2012/p/11206166.html

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号