[toc]
1 前言
设计模式一共有23种,可以分为3大类:
- 创建型模式:单例模式、抽象工厂模式、建造者模式、工厂模式、原型模式;
- 结构型模式:适配器模式、桥接模式、装饰模式、组合模式、外观模式、亨元模式、代理模式;
- 行为型模式:模板方法模式、命令模式、迭代器模式、观察者模式、中介者模式、备忘录模式、解释器模式、状态模式、策略模式、职责链模式、访问者模式;
2 面向对象设计原则
- 依赖倒置原则
- 开发封闭原则
- 单一职责原则
- 里氏替换原则(Liskov替换原则)
- 接口隔离原则
2.1 依赖倒置原则
- 高层模块(稳定)不应该依赖于底层模块(变化),二者都应该依赖于抽象(稳定);
- 抽象(稳定)不应该依赖于实现细节(变化),实现细节应该依赖于抽象(稳定);
2.2 开放封闭原则
- 对扩展开放,对更改封闭;
- 类模块应该是可扩展的,但是不可修改;
2.3 单一职责原则
- 一个类应该仅有一个引起它变化的原因;
- 变化的方向隐含着类的责任;
2.4 里氏替换原则
2.5 接口隔离原则
- 不应该强迫客户程序依赖它们不用的方法;
- 接口应该小而完备;
2.6 优先使用对象组合,而不是类继承
- 类继承通常为”白箱复用“,对象组合通常为”黑箱复用“;
- 继承在某种程度上破坏了封装性,子类父类耦合度高;
- 对象组合则只要求被组合的对象具有良好定义的接口,耦合度低;
2.7 封装变化点
- 使用封装来创建对象之间的分解层,让设计者可以在分解层一侧进行修改,而不会对另一侧产生不良的影响,从而实现层次间的松耦合;
2.8 针对接口编程,而不是针对实现编程
- 不将变量类型声明为某个特定的具体类,而是声明为某个接口;
- 客户程序无需获知对象的具体类型,只需要知道对象所具有的接口;
- 减少系统中各部分的依赖关系,从而实现“高内聚,松耦合”的类型设计方案;
3 设计模式
从可以将设计模式分为三类:
创建型模式:将对象的部分创建工作
结构型模式:
行为型模式:
3.1 模板方法模式
定义一个操作中的算法的骨架(稳定),而将一些步骤延迟(变化)到子类中。Template Method使得子类可以不改变(复用)一个算法的接口即可重定义(override 重写)该算法的某些特定步骤。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
| #include <iostream>
class Library { protected: void InputUserCmd(std::string cmd) { cmd_ = cmd; std::cout << "Library::InputUserCmd" << std::endl; }
virtual bool IsValidUserCmd(void) = 0;
void ProcessUserCmd(void) { std::cout << "Library::ProcessUserCmd" << std::endl; }
virtual void RecordLog(void) = 0;
std::string BuildCmdRsp(bool ret) { std::cout << "Library::BuildCmdRsp, result: " << ret << std::endl; if (ret) { return "Succeed"; } else { return "Fail"; } }
public: void Run(std::string cmd) { InputUserCmd(cmd);
if (IsValidUserCmd()) { ProcessUserCmd(); RecordLog(); BuildCmdRsp(true); } else { BuildCmdRsp(false); } }
std::string cmd_; };
class Application : public Library { virtual bool IsValidUserCmd(void) { std::cout << "Application::IsValidUserCmd" << std::endl; return true; }
virtual void RecordLog(void) { std::cout << "Application::RecordLog" << std::endl; } };
int main(void) { Library *app = new Application; app->Run("LOGIN IN APP.");
return 0; }
|
程序执行结果:
1 2 3 4 5
| Library::InputUserCmd Application::IsValidUserCmd Library::ProcessUserCmd Application::RecordLog Library::BuildCmdRsp, result: 1
|