设计模式基础

模式

  1. 在一定环境中解决某一问题的方案,包括三个基本元素–问题,解决方案和环境。
  2. 大白话:在一定环境下,用固定套路解决问题。

设计模式

是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计 模式是为了可重用代码、让代码更容易被他人理解、保证代 码可靠性。 毫无疑问,设计模 式于己于他人于系统都是多赢的;设计模式使代码编制真正工程化;
设计模式是软件工程的基石脉络,如同大厦的结构一样。

设计模式分类

GangofFour 的“DesignPatterns:ElementsofResualbelSoftware”书将设计模式归纳为 三大类型,共 23 种。

  1. 创建型模式 : 通常和对象的创建有关,涉及到对象实例化的方式。(共 5 种模式)
  2. 结构型模式: 描述的是如何组合类和对象以获得更大的结构。(共 7 种模式)
  3. 行为型模式: 用来对类或对象怎样交互和怎样分配职责进行描述。(共 11 种模式)

创建型模式

用来处理对象的创建过程

工厂方法模式(FactoryMethodPattern)

定义一个创建产品对象的工厂接口, 将实际创建工作推迟到子类中。

抽象工厂模式(AbstractFactoryPattern)

提供一个创建一系列相关或者相互依 赖的接口,而无需指定它们具体的类。

建造者模式(BuilderPattern)

将一个复杂的构建与其表示相分离,使得同样的 构建过程可以创建不同的表示。

原型模式(PrototypePattern

用原型实例指定创建对象的种类,并且通过拷贝这些原型创建新的对象。

单例模式(SingletonPattern)

是保证一个类仅有一个实例,并提供一个访问它的全局访问点。

结构型模式

处理类或者对象的组合

代理模式(ProxyPattern)

为其他对象提供一种代理以控制对这个对象的访问

装饰者模式(DecoratorPattern)

给一个对象添加一些额外的职责。就增加功能来 说,此模式比生成子类更为灵活。

适配器模式(AdapterPattern)

将一个类的接口转换成客户希望的另外一个接口。使得 原本由于接口不兼容而不能一起工作的那些类可以一起工作

桥接模式(BridgePattern)

将抽象部分与实际部分分离,使它们都可以独立的变化。

组合模式(CompositePattern)

将对象组合成树形结构以表示“部分–整体”的层次结 构。使得用户对单个对象和组合对象的使用具有一致性

外观模式(FacadePattern)

为子系统中的一组接口提供一个一致的界面,此模式定义 了一个高层接口,这个接口使得这一子系统更加容易使用

享元模式(FlyweightPattern)

以共享的方式高效的支持大量的细粒度的对象。

行为型模式

用来对类或对象怎样交互和怎样分配职责进行描述

模板方法模式(TemplateMethodPattern)

使得子类可以不改变一个算法的结构即可重 定义该算法的某些特定步骤。

命令模式(CommandPattern)

将一个请求封装为一个对象,从而使你可用不同的请 求对客户端进行参数化;对请求排队或记录请求日志,以及支持可撤销的操作

责任链模式(ChainofResponsibilityPattern)

在该模式里,很多对象由每一个对象对其 下家的引用而连接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理 此请求,这使得系统可以在不影响客户端的情况下动态地重新组织链和分配责任

策略模式(StrategyPattern)

是准备一组算法,并将每一个算法封装起来,使得它们 可以互换。

中介者模式(MediatorPattern)

定义一个中介对象来封装系列对象之间的交互。终 结者使各个对象不需要显示的相互调用 ,从而使其耦合性松散,而且可以独立的改变他们 之间的交互。

观察者模式(ObserverPattern)

定义对象间的一种一对多的依赖关系,当一个对象的状 态发生改变时,所有依赖于它的对象都得到通知并被自动更新。

备忘录模式(MementoPattern)

是在不破坏封装的前提下,捕获一个对象的内部状态, 并在该对象之外保存这个状态。

访问者模式(VisitorPattern)

就是表示一个作用于某对象结构中的各元素的操作,它使 你可以在不改变各元素的类的前提下定义作用于这些元素的新操作。

状态模式(StatePattern)

对象的行为,依赖于它所处的状态

解释器模式(InterpreterPattern)

描述了如何为简单的语言定义一个语法,如何在 该语言中表示一个句子,以及如何解释这些句子。

迭代器模式(IteratorPattern)

提供了一种方法顺序来访问一个聚合对象中的各个元 素,而又不需要暴露该对象的内部表示。

设计模式基本原则

最终目的:高内聚,低耦合

开放封闭原则 (OCP,OpenForExtension,ClosedForModificationPrinciple)

类的改动是通过增加代码进行的,而不是修改源代码。

#include<iostream>using namespace std;
//开闭原则//写一个抽象类
class AbstractCaculator
{
public:virtual int getResult() = 0;virtual void setOperatorNumber(int a, int b) = 0;};//加法计算器
class PlusCaculator :public AbstractCaculator{
public:virtual void setOperatorNumber(int a, int b){this->mA = a;this->mB = b;}virtual int getResult(){return mA + mB;}
private:int mA;int mB;
};//减法计算器
class MinuteCaculator :public AbstractCaculator{
public:virtual void setOperatorNumber(int a, int b){this->mA = a;this->mB = b;}virtual int getResult(){return mA - mB;}
private:int mA;int mB;
};//乘法计算器
class MultiplyCaculator :public AbstractCaculator{
public:virtual void setOperatorNumber(int a, int b){this->mA = a;this->mB = b;}virtual int getResult(){return mA * mB;}
private:int mA;int mB;
};void test01()
{//父类指针指向基类对象AbstractCaculator * caculator = new PlusCaculator;caculator->setOperatorNumber(10, 20);cout << "ret:" << caculator->getResult() << endl;caculator = new  MultiplyCaculator;caculator->setOperatorNumber(30,20);cout << "ret:" << caculator->getResult() << endl;
}int main(void)
{test01();system("pause");return 0;
}

我们如果想再次增加计算器类型,只用继承基类,重写方法就可以了。

单一职责原则 (SRP,SingleResponsibilityPrinciple)

类的职责要单一,对外只提供一种功能,而引起类变化的原因都应该只有一个。

依赖倒置原则 (DIP,DependenceInversionPrinciple)

依赖于抽象(接口),不要依赖具体的实现(类),也就是针对接口编程。

#include<iostream>
using namespace std;class BankWorker{
public:void SaveService(){cout << "办理存款业务..." << endl;}void payService(){cout << "办理支付业务.." << endl;}void tranferService(){cout << "办理转账业务..." << endl;}
};void doSaveBussiness(BankWorker *worker){worker->SaveService();
}
void doPayBussiness(BankWorker *worker){worker->payService();
}
void doTransferBussiness(BankWorker *worker){worker->tranferService();
}void test01(){BankWorker *worker = new BankWorker;doSaveBussiness(worker);//办理存款业务doPayBussiness(worker);//办理支付业务doTransferBussiness(worker);//办理转账业务
}int main(void)
{system("pause");return 0;
}

//银行工作人员
class AbstractWotker{
public:virtual void doBussiness() = 0;};//专门负责办理存款业务的工作人员
class doSaveBankWorker :public AbstractWotker{
public:virtual void doBussiness(){cout << "办理存款业务.." << endl;}
};class PaySaveBankWorker :public AbstractWotker{
public:virtual void doBussiness(){cout << "办理支付业务.." << endl;}
};class TransferSaveBankWorker :public AbstractWotker{
public:virtual void doBussiness(){cout << "办理转账业务.." << endl;}
};//中层业务,依赖于抽象层,bu
void doNewBusiness(AbstractWotker * worker){worker->doBussiness();delete worker;
}void test02()
{doNewBusiness(new TransferSaveBankWorker);doNewBusiness(new doSaveBankWorker);doNewBusiness(new PaySaveBankWorker);
}int main(void)
{//test01();system("pause");return 0;
}

接口隔离原则 (ISP,InterfaceSegegationPrinciple)

不应该强迫客户的程序依赖他们不需要的接口方法。一个接口应该只提供一种对外功能, 不应该把所有操作都封装到一个接口中去。

里氏替换原则 (LSP,LiskovSubstitutionPrinciple)

任何抽象类出现的地方都可以用他的实现类进行替换。实际就是虚拟机制,语言级别实 现面向对象功能。

合成复用原则(CARP,Composite/AggregateReusePrinciple)

优先使用组合而不是继承原则。如果使用继承,会导致父类的任何变换都可能影响到子类的行为。 如果使用对象组合,就降低了这种依赖关系。

#include<iostream>
using namespace std;class AbstractCar{
public:virtual void run() = 0;};class Dazhong :public AbstractCar{
public:virtual void run(){cout << "大众车启动..." << endl;}
};class Tuolaji :public AbstractCar{
public:virtual void run(){cout << "拖拉机启动.." << endl;}
};//针对具体类,不使用继承
#if 0
class Person :public Tuolaji{
public:void Doufeng(){run();}
};class PersonB :public Tuolaji{
public:void Doufeng(){run();}
};
#endif//用组合
class Person{
public:/*~Person(){if (this->car != NULL){delete this->car;}}*/void setCar(AbstractCar *car){this->car = car;}void Doufeng(){this->car->run();if (this->car != NULL){delete this->car;this->car = NULL;}}public:AbstractCar *car;
};void test02(){Person* p = new Person;p->setCar(new Dazhong);p->Doufeng();p->setCar(new Tuolaji);p->Doufeng();delete p;
}//继承和组合优先使用组合
int main()
{test02();system("pause");return 0;
}

迪米特法则(LOD,LawofDemeter)

一个对象应当对其他对象尽可能少的了解,从而降低各个对象之间的耦合,提高系统的 可维护性。例如在一个程序中,各个模块之间相互调用时,通常会提供一个统一的接口来实 现。这样其他模块不需要了解另外一个模块的内部实现细节,这样当一个模块内部的实现发 生改变时,不会影响其他模块的使用。

#include<iostream>
#include<string>
#include<vector>
using namespace std;
//迪米特原则,又叫最小知识原则,不要暴露内部结构,只提供接口class AbstractBuiding
{
public:virtual void sale() = 0;virtual string getQuality() = 0;
};//楼盘A
class BuildingA :public AbstractBuiding
{
public:BuildingA(){mQuity = "高品质";}virtual void sale(){cout << "楼盘A" << mQuity << "被售卖";}virtual string getQuality(){return mQuity;}
public:string mQuity;
};//楼盘B
class BuildingB :public AbstractBuiding
{
public:BuildingB(){mQuity = "低品质";}virtual void sale(){cout << "楼盘B" << mQuity << "被售卖";}virtual string getQuality(){return mQuity;}
public:string mQuity;
};
#if 0
void test01()
{BuildingA * ba = new BuildingA;if (ba->mQuity == "低品质"){ba->sale();}BuildingB *bb = new BuildingB;if (bb->mQuity == "低品质"){bb->sale();}
}
#endif//中介类
class Mediator{
public:Mediator(){AbstractBuiding * building = new BuildingA;vBuilding.push_back(building);building = new BuildingB;vBuilding.push_back(building);}~Mediator(){for (vector<AbstractBuiding*>::iterator it = vBuilding.begin(); it != vBuilding.end(); it++){if (*it != NULL){delete *it;}}}//对外提供接口AbstractBuiding *findMyBuilding(string qulity){for (vector<AbstractBuiding *>::iterator it = vBuilding.begin(); it != vBuilding.end(); it++){if ((*it)->getQuality() == qulity){return *it;}}return NULL;}
public:vector<AbstractBuiding*>vBuilding;
};void test02()
{Mediator *mediator = new Mediator;AbstractBuiding*building = mediator->findMyBuilding("高品质");if (building != NULL){building->sale();}else{cout << "没有符合您条件的楼盘!" << endl;}
}int main(void)
{//test01();test02();system("pause");return 0;
}

就是有个中间商。

设计模式--1(设计模式基础,设计模式基本原则,设计模式分类)相关推荐

  1. 设计模式学习总结(一)——设计原则与UML统一建模语言

    目录 一.概要 1.1.设计模式定义 1.2.设计模式分类 1.3.设计模式书籍 二.UML统一建模语言 2.1.UML分类 2.2.类图 2.2.1.关联 2.2.2.聚合/组合 2.2.3.依赖 ...

  2. python算法基础设计模式,python常见的设计模式

    Python有设计模式么 Python设计模式主要分为三大类:创建型模式.结构型模式.行为型模式;三 大类中又被细分为23种设计模式,以下这几种是最常见的. 单例模式:是一种常用的软件设计模式,该模式 ...

  3. 【设计模式学习01】设计模式概述,UML图,软件设计原则

    文章目录 1. 设计模式概述 1.1 软件设计模式的产生背景 1.2 软件设计模式的概念 1.3 学习设计模式的必要性 1.4 设计模式分类 2. UML图 2.1 类图概述 2.2 类图的作用 2. ...

  4. 设计模式01 UML图,软件设计原则,创建型模式

    概述 "设计模式"最初并不是出现在软件设计中,而是被用于建筑领域的设计中. 1995年,由 Erich Gamma.Richard Helm.Ralph Johnson 和 Joh ...

  5. 软件设计模式--第一章 软件设计模式基础

    目录 第一章 软件设计模式基础 1.软件设计模式概述 (1)什么是软件设计模式 (2)学习设计模式的意义 (3)软件设计模式的基本要素 (4)GoF的23种设计模式简介 2.UML中的类图 (1)统一 ...

  6. php注册树模式,php基础设计模式大全(注册树模式、工厂模式、单列模式)

    废话不多说了,先给大家介绍注册树模式然后介绍工厂模式最后给大家介绍单列模式,本文写的很详细,一起来学习吧. php注册树模式 什么是注册树模式? 注册树模式当然也叫注册模式,注册器模式.之所以我在这里 ...

  7. Checking Table 设计模式 - 从概念、建模、设计到实现

    如何基于业务需求驱动理念来开展我们的模式创新,成为了当今架构师.设计师的重要职责之一.本文通过具体的 Checking Table 设计模式案例创新过程,阐述在核心业务需求分析中如何开展建模.设计并实 ...

  8. Checking Table 设计模式 - 从概念、建模、设计到实现——兼谈基于业务需求驱动的设计模式创新

    郑 先全, 架构师, NEC Asia Pacific Pte Ltd 简介: 如何基于业务需求驱动理念来开展我们的模式创新,成为了当今架构师.设计师的重要职责之一.本文通过具体的 Checking ...

  9. 二、设计模式-必要的基础知识—旅行前的准备 #和设计模式一起旅行#

    必要的基础知识-旅行前的准备 工欲善其事,必先利其器.--<论语> 要开始一场旅行,准备的工作肯定不能少,不能太任性,一场说走就走的旅行,也需要基础条件的,那么本次就做做一些旅行之前的准备 ...

最新文章

  1. 14 款命令行常用工具的替代品!
  2. [安卓基础] 006.打开另一个Activity
  3. DL框架之darknet:深度学习框架darknet的简介、安装、使用方法的详细攻略
  4. Servlet程序入门
  5. P4316-绿豆蛙的归宿【数学期望】
  6. day32 管道, 数据共享, 进程池, 回调函数
  7. python函数做n_【python】定义函数、参数、递归(n!)
  8. 关于用C#编写ActiveX控件2(转)
  9. SQL Server高级查询之T-SQL编程(存储过程)
  10. HDOJ1018 ( Big Number ) 【斯特林公式---处理阶乘及阶乘位数的问题】
  11. Ubuntu16.04下Nvidia+Cuda8.0+Dynet安装教程
  12. 【GCC调试程序C语言问题】对‘ceil’未定义的引用/已经包含头文件仍然提示未定义的引用
  13. android游戏开发方向初探
  14. Windows删除服务
  15. 音视频篇 - Android 音视频涉及到的技术
  16. acwing1148——秘密的牛奶运输(求次小生成树)
  17. Web网站模板-教育培训响应式网站模板(HTML+CSS+JavaScript)
  18. 2023北京老博会,CISSE中国国际养老服务业博览会
  19. Android 距离传感器修复 修复打电话黑屏 无法快速唤醒屏幕的BUG
  20. linux2017期末试卷,LINUX认证考试模拟试题及答案

热门文章

  1. 2019 The 19th Zhejiang University Programming Contest
  2. 配置Ubuntu虚拟环境
  3. 显示日历的指令:cal
  4. SharePoint2013安装组件时AppFabric时出现1603错误,解决方法:
  5. jquery validation-jquery的验证框架 详解(1)
  6. Linux中的Ramdisk和Initrd
  7. 剑灵系统推荐加点_剑灵重制修炼系统 无定式加点打造自我风格
  8. mysql序列号生成_一文看懂mycat的6种全局序列号实现方式
  9. java方法调用机制_Java方法调用机制 - osc_bkdv2it5的个人空间 - OSCHINA - 中文开源技术交流社区...
  10. 求10以内平均数的c语言,求助 给小学生出题,自己选加减乘除 做10题 10以内的数 然后统计分...