当前位置: 首页 > news >正文

[架构设计] 结构型模型

目录

一、代理模式

二、装饰模式

三、外观模式

四、适配器模式


一、代理模式

通过一个代理类,来控制对这个对象的访问。

类别描述
适用场景
  • 为其他对象提供一种代理以控制对这个对象的访问
优点
  • 能够协调调用者和被调用者,在一定程度上降低了系统的耦合度
缺点
  • 代理实现较为复杂

无代理模式,人人可具备权限

#include <iostream>
using namespace std;


//提供一种代理来控制对其他对象的访问

class AbstraactCommonInterface {
public:
	virtual void run() = 0;

};

//我已经写好的系统
class Mysystem {
public:
	virtual void run() {
		cout << "系统启动..." << endl;
	}
};

//必须有权限验证,不是所有人都能来启动我的启动,必须提供用户名和密码

int main() {
	//这样不行。是个人都能启动
	Mysystem* system = new Mysystem;
	system -> run();

	return 0;
}

增加代理模式

#include <iostream>
using namespace std;

//提供一种代理来控制对其他对象的访问
class AbstraactCommonInterface {
public:
	virtual void run() = 0;

};

//我已经写好的系统
class Mysystem :public AbstraactCommonInterface{
public:
	virtual void run() {
		cout << "系统启动..." << endl;
	}
};

//必须有权限验证,不是所有人都能来启动我的启动,必须提供用户名和密码
class MysystemProxy :public AbstraactCommonInterface {
public:
	MysystemProxy(string username,string password) {
		this->mUsername = username;
		this->mPassword = mPassword;
		
	}
	bool checkUsernameAndPassword() {
		if (mUsername == "admin"&&mPassword == "admin") {
			return true;
		}
		return false;
	}
	virtual void run() {
		if (checkUsernameAndPassword()) {
			cout << "用户名和密码正确,验证通过..." << endl;
			this->pSystem->run();
		}
		else {
			cout << "用户名或密码错误,权限不足...." << endl;
		}
	}
	~MysystemProxy() {
		if (pSystem != NULL) {
			delete pSystem;
		}
	}
public:
	Mysystem* pSystem;
	string mUsername;
	string mPassword;
};


int main() {
#if 0
	//这样不行。是个人都能启动
	Mysystem* system = new Mysystem;
	system -> run();
#endif
	//调用代理模式
	MysystemProxy* proxy = new MysystemProxy("root","admin");
	proxy->run();

	return 0;
}

二、装饰模式

通过一种对客户端透明的方式,扩展对象功能

具体操作是,功能放到单独的类中,客户端可以选择、排序等装饰对象。

类别描述
适用场景
  • 动态、透明的方式给单个对象添加职责
  • 当不能采用继承的方式对系统进行扩展或者采用继承不利于系统扩展和维护时可以使用装饰模式
优点
  • 装饰模式比继承更加灵活性,不会导致类的个数急剧增加
  • 通过一种动态的方式来扩展一个对象的功能,从而实现不同的行为
  • 可以对一个对象进行多次装饰
缺点
  • 装饰模式进行系统设计时将产生很多小对象,大量小对象的产生势必会占用更多的系统资源,影响程序的性能
#include <iostream>
using namespace std;

//一般情况下,用继承实现类的功能拓展
//装饰模式 可以动态给一个类增加功能

//抽象英雄
class AbstractHero {
public:
	virtual void ShowStatus() = 0;
public:
	int mHp;
	int mMp;
	int mAt;
	int mDf;
};

//具体英雄
class HeroA :public AbstractHero {
public:
	HeroA() {
		mHp = 0;
		mMp = 0;
		mAt = 0;
		mDf = 0;
	}
	virtual void ShowStatus() {
		cout << "血量:" << mHp << endl;
		cout << "魔法:" << mMp << endl;
		cout << "攻击:" << mAt << endl;
		cout << "防御:" << mDf << endl;
	}
};

//英雄穿上某个装饰物 那么他还是个英雄
//装饰物
class AbstractEquipmet : public AbstractHero {
public:
	AbstractEquipmet(AbstractHero* hero) {
		this->pHero = hero;
	}
	virtual void ShowStatus() = 0;
public:
	AbstractHero* pHero;

};

//狂徒
class KuangtuEquipment :public AbstractEquipmet {
public:
	KuangtuEquipment(AbstractHero* hero) :AbstractEquipmet(hero) {}
	//增加额外的功能
	void AddKuangtu() {
		cout << "英雄穿上狂徒之后..." << endl;
		this->mHp = this->pHero->mHp;
		this->mMp = this->pHero->mMp;
		this->mAt = this->pHero->mAt;
		this->mDf = this->pHero->mDf + 30;

		delete this->pHero;
	}
	virtual void ShowStatus() {
		AddKuangtu();
		cout << "血量:" << mHp << endl;
		cout << "魔法:" << mMp << endl;
		cout << "攻击:" << mAt << endl;
		cout << "防御:" << mDf << endl;
	}
};

//无尽
class Wujing : public AbstractEquipmet {
public:
	Wujing(AbstractHero* hero) :AbstractEquipmet(hero) {}
	//增加额外的功能
	void AddWujing() {
		cout << "英雄穿上无尽之后..." << endl;
		this->mHp = this->pHero->mHp;
		this->mMp = this->pHero->mMp;
		this->mAt = this->pHero->mAt + 80;
		this->mDf = this->pHero->mDf;

		delete this->pHero;
	}
	virtual void ShowStatus() {
		AddWujing();
		cout << "血量:" << mHp << endl;
		cout << "魔法:" << mMp << endl;
		cout << "攻击:" << mAt << endl;
		cout << "防御:" << mDf << endl;
	}
};


int main() {
	AbstractHero* hero = new HeroA;
	hero->ShowStatus();
	cout << "----------------------------" << endl;

	//给裸奔的英雄穿上衣服后
	hero = new KuangtuEquipment(hero);
	hero->ShowStatus();

	cout << "----------------------------" << endl;

	//装备武器
	hero = new Wujing(hero);
	hero->ShowStatus();

	return 0;
}

三、外观模式

两个子系统,如果都初始化,传统需要一个一个new一下,现在提供一个类,在这个类里面做完初始化工作。

外观模式就是将复杂的子类系统抽象到同一个的接口进行管理,外界只需要通过此接口与子类系统进行交互,而不必要直接与复杂的子类系统进行交互

类别描述
适用场景
  • 复杂系统需要简单入口使用
  • 客户端程序与多个子系统之间存在很大的依赖性
  • 可以使用外观模式定义系统中每一层的入口,层与层之间不直接产生联系,而通过外观类建立联系,降低层之间的耦合度
优点
  • 屏蔽了子系统组件,减少了客户端所需处理的对象数目,并使得子系统使用起来更加容易
  • 子系统与客户端之间的松耦合关系,这使得子系统的变化不会影响到调用它的客户端,只需要调整外观类即可
  • 子系统的修改对其他子系统没有任何影响
缺点
  • 对客户端访问子系统类做太多的限制则减少了可变性和灵活性
  • 设计不当,增加新的子系统可能需要修改外观类的源代码
#include <bits/stdc++.h>
using namespace std;

//电视机
class Televison {
public:
	void On() {
		cout << "电视机打开..." << endl;
	}
	void Off() {
		cout << "电视机关闭" << endl;
	}
};
//灯
class Light {
public:
	void On() {
		cout << "灯打开..." << endl;
	}
	void Off() {
		cout << "灯关闭" << endl;
	}
};
//音箱
class Audio {
public:
	void On() {
		cout << "音箱打开..." << endl;
	}
	void Off() {
		cout << "音箱关闭" << endl;
	}
};
//麦克风
class Microphone {
public:
	void On() {
		cout << "麦克风打开..." << endl;
	}
	void Off() {
		cout << "麦克风关闭" << endl;
	}
};
//DVD
class DVDPlayer {
public:
	void On() {
		cout << "DVD播放器打开..." << endl;
	}
	void Off() {
		cout << "DVD播放器关闭" << endl;
	}
};
//游戏机
class Gamemachine {
public:
	void On() {
		cout << "游戏机打开..." << endl;
	}
	void Off() {
		cout << "游戏机关闭" << endl;
	}
};

//KTV模式
class KTVModel {
public:
	KTVModel() {
		pTv = new Televison;
		pLight = new Light;
		pAudio = new Audio;
		pMicrophone = new Microphone;
		pDVD = new DVDPlayer;
	}

	void OnKtv() {
		pTv->On();
		pLight->Off();
		pAudio->On();
		pMicrophone->On();
		pDVD->On();
	}

	void OffKtv() {
		pTv->Off();
		pLight->On();
		pAudio->Off();
		pMicrophone->Off();
		pDVD->Off();
	}

	~KTVModel() {
		delete pTv;
		delete pLight;
		delete pAudio;
		delete pMicrophone;
		delete pDVD;
	}

public:
	Televison* pTv;
	Light* pLight;
	Audio* pAudio;
	Microphone* pMicrophone;
	DVDPlayer* pDVD;
};

int main() {
	KTVModel* ktv = new KTVModel;
	ktv->OnKtv();

	return 0;
}

四、适配器模式

使得原本由于接口不兼容而不能一起工作的那些类可以一起工作

类别描述
适用场景
  • 现有的类,而这些类的接口(如方法名)不符合系统的需要,甚至没有这些类的源代码
  • 创建一个可以重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作
优点
  • 目标类和适配者类解耦,通过引入一个适配器类来重用现有的适配者类,无须修改原有结构
  • 客户端类而言是透明的,而且提高了适配者的复用性,同一个适配者类可以在多个不同的系统中复用
  • 可以在不修改原有代码的基础上增加新的适配器类
缺点
  • 适配器中置换适配者类的某些方法比较麻烦
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;

//适配器模式 就是将已经写好的接口,但是这个接口不符合需求
//将写好的接口转换成目标接口

//这函数我已经写好
struct Myprint{
	void operator()(int v1,int v2) {
		cout << v1 + v2 << endl;
	}
};

//定义目标接口 我要是配偶 适配成什么样的,
//要适配成只能传一个参数的,适配for_each第三个参数的适用
class Target {
public:
	virtual void operator()(int v) = 0;

};

//写适配器
class Adapater :public Target {
public:
	Adapater(int param) {
		this->param = param;
	}
	virtual void operator() (int v) {
		print(v,param);
	}
public:
	Myprint print;
	int param;
};

//MyBind2nd,原来param固定的10,现在提供一个方法改
Adapater MyBind2nd(int v) {
	return Adapater(v);
}


int main(void) {
	vector<int> v;
	for (int i = 0; i < 10; i++) {
		v.push_back(i);
	}
	
	//适配器模式的运用
	//for_each()的第三个参数是个带一个参数的函数,但是Myprint需要两个参数
	for_each(v.begin(),v.end(), MyBind2nd(10));

	return 0;
}

相关文章:

  • file | 某文件夹【解耦合】下的文件查找功能实现及功能单元测试
  • 每日刷题(图论)
  • [Leetcode 216][Medium]组合总和 III--回溯
  • 如何实现OpenHarmony的OTA升级
  • 回归预测|基于卷积神经网络-鲸鱼优化-最小二乘支持向量机的数据回归预测Matlab程序 CNN-WOA-LSSVM
  • 实习项目|苍穹外卖|day1
  • sql | leecode 1147 |即时事务配送II | sql 优化
  • [数据集][目标检测]狗种类检测数据集VOC+YOLO格式20578张120类别
  • 常用软件下载地址
  • Go语言反射(reflect)
  • Spring-Cloud-Gateway集成Sentinel限流
  • 【已解决】用ArcGIS处理过的数据在QGIS中打开发生偏移怎么办?| 数据在ArcGIS中打开位置正常,在QGIS中偏移
  • [附源码]计算机毕业设计基于springboot的汽车租赁系统
  • [附源码]Python计算机毕业设计SSM竞赛报名管理系统(程序+LW)
  • mssql(1433端口)介绍
  • 文华财经期货傻瓜式操作设置期货止盈止损指标公式,期货技术分析多空平仓离场信号
  • java计算机毕业设计医院挂号管理系统源程序+mysql+系统+lw文档+远程调试
  • Springboot流浪动物管理系统p2326计算机毕业设计-课程设计-期末作业-毕设程序代做
  • Dreamweaver网页设计与制作100例 餐饮主题简洁日式料理餐饮网页设计(4页)HTML+CSS+JavaScript
  • [附源码]计算机毕业设计点餐系统
  • 海口市美兰区图书馆建筑结构设计(计算书+任务书+建筑结构施工组织设计cad图纸)
  • 昨晚停网后,我写了一段Python代码攻破了隔壁老王家的wifi密码
  • [激光原理与应用-36]:《光电检测技术-3》- 光学测量基础 - 光电效应与光电探测器的基本原理
  • 计算机毕业设计Java电子病历系统(源码+系统+mysql数据库+lw文档)
  • 【LIN总线测试】——LIN主节点数据链路层测试
  • Docker consul的容器服务更新与发现
  • [附源码]计算机毕业设计小区物业管理系统Springboot程序
  • 数据结构与算法,MySQL数据库面试专题及答案
  • 一篇文章学会调优 ClickHouse
  • (41)Verilog实现递归功能
  • 【数据结构1】数据结构的基本概念
  • 【通信原理】数字基带传输的线路码型