C++11 设计模式7 策略模式 ,Strategy

策略模式的概念:

策略模式(Strategy Pattern)是 C++ 中常用的一种行为设计模式,它能在运行时改变对象的行为。在策略模式中,一个类的行为或其算法可以在运行时更改。这种类型的设计模式属于行为模式。

在策略模式中,需要创建表示各种策略的对象和一个行为随着策略对象改变而改变的 Context 对象。策略对象更改 Context 对象的执行算法。

在策略模式中,通常包括以下几个角色:

(1)策略接口(Strategy Interface): 定义了一个策略的公共接口,所有具体的策略类都需要实现这个接口。这个接口声明了策略对象将执行的操作。

(2)具体策略类(Concrete Strategy Classes): 实现了策略接口,提供了具体的算法或行为。每个具体策略类都封装了实现特定行为或算法的代码。

(3)上下文(Context): 维护一个指向策略对象的引用,并定义一个接口来让策略对象执行其算法。上下文类并不知道具体的策略类,它只知道策略接口。这样,上下文可以将请求转发给当前关联的策略对象来执行。

2 策略模式的实现步骤
在 C++ 实现策略模式的实现步骤如下:

(1)定义策略接口:
首先,需要定义一个策略接口。这个接口通常是一个纯虚类,声明了一组公共的、需要由具体策略类来实现的方法。这些方法是策略对象将执行的行为的抽象描述。

(2)实现具体策略类:
接下来,创建实现策略接口的具体策略类。每个具体策略类都包含了实现特定算法或行为的代码。这些类继承了策略接口,并实现了接口中声明的所有方法。

(3)创建上下文类:
上下文类负责维护对策略对象的引用,并定义了一个接口,以便客户端代码可以通过这个接口来执行策略对象的方法。上下文类通常包含一个指向策略接口的指针或引用,并通过这个指针或引用来调用策略方法。上下文类本身并不关心具体使用了哪个策略,它只关心策略接口。

(4)在上下文中设置策略对象:
在客户端代码中,创建具体策略类的对象,并将其传递给上下文对象。上下文对象使用这个策略对象来执行相应的算法或行为。客户端代码可以通过调用上下文类的方法来间接调用策略对象的方法。

(5)执行策略:
客户端代码通过调用上下文类的执行方法(例如 executeStrategy()),来触发策略的执行。上下文类将调用当前设置的策略对象的方法,实现相应的算法或行为。

(6)更改策略:
如果需要改变行为,客户端代码可以创建另一个策略对象,并将其设置为上下文对象的新策略。这样,上下文对象在执行策略时会使用新的算法或行为。

通过这些步骤,策略模式允许在运行时动态地改变对象的行为,提高了代码的灵活性和可维护性。它通过将算法和行为封装在独立的策略类中,实现了算法与使用算法的客户端代码之间的解耦。这样,客户端代码只需关注于如何使用策略,而不需要关心策略的具体实现。

#include <iostream>  
#include <memory>

// 步骤1: 定义策略接口  
class Strategy {
public:
	virtual ~Strategy() {}
	virtual void execute() = 0;
};

// 步骤2: 实现具体策略类  
class ConcreteStrategyA : public Strategy {
public:
	void execute() override {
		std::cout << "Executing strategy A" << std::endl;
	}
};

class ConcreteStrategyB : public Strategy {
public:
	void execute() override {
		std::cout << "Executing strategy B" << std::endl;
	}
};

// 步骤3: 创建上下文类  
class Context {
public:
	// 通过构造函数设置策略  
	Context(std::unique_ptr<Strategy> strategy) : strategy(std::move(strategy)) {}

	// 执行策略  
	void executeStrategy() {
		if (strategy) {
			strategy->execute();
		}
		else {
			std::cout << "No strategy is set." << std::endl;
		}
	}

	// 更改策略  
	void setStrategy(std::unique_ptr<Strategy> newStrategy) {
		strategy = std::move(newStrategy);
	}

private:
	std::unique_ptr<Strategy> strategy; // 使用unique_ptr管理策略对象的生命周期  
};

// 步骤4-6: 在客户端代码中设置和执行策略  
int main() 
{
	// 创建具体策略对象并使用unique_ptr管理  
	auto strategyA = std::make_unique<ConcreteStrategyA>();
	auto strategyB = std::make_unique<ConcreteStrategyB>();

	// 创建上下文对象并设置初始策略  
	Context context(std::move(strategyA));
	context.executeStrategy(); // 输出:Executing strategy A  

	// 更改策略  
	context.setStrategy(std::move(strategyB));
	context.executeStrategy(); // 输出:Executing strategy B  

	// 此时strategyA和strategyB已经被unique_ptr自动释放  

	return 0;
}

假设公司是有一个算税法的函数。

不同的国家有不同的算税法的逻辑。

刚开始有三个国家,china,us,japan

你可能这么写:

//不同的国家有不同的算税法的逻辑。各个国家会根据不同销售额计算税率
//
//刚开始有三个国家,china,us,japan
//
//你可能这么写:

#include <iostream>
using namespace std;

enum TaxCountry {
	china_tax,
	us_tax,
	japan_tax
};

//销售额类
class SaleOrder {
public:
	SaleOrder(int oneyeassalenum) {
		onesalemoney = oneyeassalenum;
	}

	int calculateTax() {
		if (taxcoutury== china_tax) {
			//如果是中国
			return onesalemoney / 3;
		}
		else if (taxcoutury == us_tax) {
			return onesalemoney / 2;
		}
		else if (taxcoutury == japan_tax) {
			return onesalemoney / 4;
		}
	}

	void setTaxCountry(TaxCountry _taxcoutury) {
		taxcoutury = _taxcoutury;
	}

private:
	TaxCountry taxcoutury;
	int onesalemoney;
};

int main()
{
    std::cout << "Hello World!\n";
	SaleOrder so(100);
	so.setTaxCountry(TaxCountry::china_tax);
	int result = so.calculateTax();
	cout << "result = " << result << endl;
}

但是随着业务的扩展,我们可能还要做德国的,法国的,印度的

那么这就要改动代码了,不符合开闭原则。

使用 策略模式改动

// 005策略模式.cpp : 此文件包含 "main" 函数。程序执行将在此处开始并结束。
//假设我们是做
//不同的国家有不同的算税法的逻辑。各个国家会根据不同销售额计算税率
//
//刚开始有三个国家,china,us,japan
//
//你可能这么写:

#include <iostream>
using namespace std;

enum TaxCountry {
	china_tax,
	us_tax,
	japan_tax
};

//销售额类
class SaleOrder {
public:
	SaleOrder(int oneyeassalenum) {
		onesalemoney = oneyeassalenum;
	}

	int calculateTax() {
		if (taxcoutury== china_tax) {
			//如果是中国
			return onesalemoney / 3;
		}
		else if (taxcoutury == us_tax) {
			return onesalemoney / 2;
		}
		else if (taxcoutury == japan_tax) {
			return onesalemoney / 4;
		}
	}

	void setTaxCountry(TaxCountry _taxcoutury) {
		taxcoutury = _taxcoutury;
	}

private:
	TaxCountry taxcoutury;
	int onesalemoney;
};

int main005()
{
    std::cout << "Hello World!\n";
	SaleOrder so(100);
	so.setTaxCountry(TaxCountry::china_tax);
	int result = so.calculateTax();
	cout << "result = " << result << endl;

	return 0;
}



// 策略基类
class TaxStrategy {
public:
	virtual int calculateTax(int onesalemoney) = 0;
	virtual ~TaxStrategy() {}
};

// china策略实现
class ChinaTaxStrategy : public TaxStrategy {
public:
	int calculateTax(int onesalemoney) override {

		cout << "china calculateTax" << endl;
		return onesalemoney / 3;
	}
};

// us 策略实现
class USTaxStrategy : public TaxStrategy {
public:
	int calculateTax(int onesalemoney) override {

		cout << "us calculateTax" << endl;
		return onesalemoney / 2;
	}
};

//japan 策略实现
class JapanTaxStrategy : public TaxStrategy {
public:
	int calculateTax(int onesalemoney) override {

		cout << "japan calculateTax" << endl;
		return onesalemoney / 4;
	}
};

//法国策略实现
class FranceTaxStrategy : public TaxStrategy {
public:
	int calculateTax(int onesalemoney) override {

		cout << "france calculateTax" << endl;
		return onesalemoney / 5;
	}
};

// 上下文类
class Context {
private:
	TaxStrategy* strategy;
	int oneyearmoney;
public:
	Context(TaxStrategy* s, int money) : strategy(s), oneyearmoney(money){}

	int ContextInterface() {
		return strategy->calculateTax(oneyearmoney);
	}
};


int main() {

	_CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);//程序退出时检测内存泄漏并显示到“输出”窗口

	TaxStrategy * ts = NULL;
	ts = new ChinaTaxStrategy();

	int result  = ts->calculateTax(1000);
	cout << "result = " << result << endl;

	delete ts;



	//	也可以通过创建一个上下文来处理
		// 创建具体策略对象
	TaxStrategy* strategyA = new ChinaTaxStrategy();
	TaxStrategy* strategyB = new USTaxStrategy();

	// 设置策略并调用
	Context* contextChina = new Context(strategyA,10000);
	result = contextChina->ContextInterface();

	cout << "result = " << result << endl;


	Context* contextUS = new Context(strategyB,200000);
	result = contextUS->ContextInterface();

	cout << "result = " << result << endl;

	delete contextUS;
	delete contextChina;
	delete strategyB;
	delete strategyA;

	return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/601007.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

初始C++(一)

目录 前言&#xff1a; 命名空间&#xff1a; 总结&#xff1a; 前言&#xff1a; C语言学好了&#xff0c;现在当然要进阶了&#xff0c;那么就是从C开始。 C兼容C&#xff0c;支持其中90%的语法。可能有很多同学听说过C#&#xff0c;C#和C没有关系&#xff0c;是微软研究出…

SD-WAN对云服务的优化

在云服务日益普及的当下&#xff0c;SD-WAN技术正成为众多企业优化网络连接的首选方案。其通过优化云集成和连接&#xff0c;以及增强应用程序性能&#xff0c;为企业带来了前所未有的业务效益。这种革新性的云连接方式极大地促进了企业对全球劳动力和潜在客户的触达能力。 软件…

Java中Maven的依赖管理

依赖介绍 是指当前项目运行所需要的jar包&#xff0c;一个项目中可以引入多个依赖 配置 在pom.xml中编写<dependencies>标签 在<dependencies>中使用<dependency>引入标签 定义坐标的groupId、rtifactId、version 点击刷新按钮、引入新坐标 例如引入下…

git bash各分支修改内容不同但合并后不显示冲突问题

在跟着廖雪峰老师的git学习时&#xff0c;按部就班的执行明后&#xff0c;发现 而不是出现原文的结果 解决方法&#xff1a; 切换位feature分支&#xff0c;再合并 git switch feature1 git merge master 此时我们发现&#xff1a; 后面再跟着原文敲就可以了

【spark RDD】spark 之 Kryo高性能序列化框架

文章目录 一. RDD序列化的原因二. Kryo序列化框架三. spark 配置 kryo 序列化1. 设定kryo序列化2. 注册序列化类&#xff08;非必须&#xff0c;但是强烈建议做&#xff09;3. 配置 spark.kryoserializer.buffer 一. RDD序列化的原因 Spark初始化工作是在Driver端进行的&#…

机器人系统ros2-开发实践07-将机器人的状态广播到 tf2(Python)

上个教程将静态坐标系广播到 tf2&#xff0c;基于这个基础原理这个教程将演示机器人的点位状态发布到tf2 1. 写入广播节点 我们首先创建源文件。转到learning_tf2_py我们在上一教程中创建的包。在src/learning_tf2_py/learning_tf2_py目录中输入以下命令来下载示例广播示例代码…

【机器学习与实现】线性回归示例——波士顿房价分析

目录 一、创建Pandas对象并查看数据的基本情况二、使用皮尔逊相关系数分析特征之间的相关性三、可视化不同特征与因变量MEDV&#xff08;房价中值&#xff09;间的相关性四、划分训练集和测试集并进行回归分析 一、创建Pandas对象并查看数据的基本情况 boston.csv数据集下载&a…

Grotesque系列靶机Grotesque1

第一步信息收集&#xff1a; 靶机ip&#xff1a;192.168.108.131 攻击机IP&#xff1a;192.168.108.128 nmap扫描靶机的可用端口&#xff1a; 发现http服务的端口存在66端口和80端口 扫描一下靶机端口的http服务&#xff1a; 可以看到&#xff0c;默认的80端口是不存在的&am…

25_Scala集合Tuple

文章目录 tuple1.元组定义2.Tuple元素访问3.如果元素的len2&#xff0c;称之为键值对对象&#xff0c;也称之为对偶元组4.补充上节Map5.Map集合遍历6.集合之间相互转化 tuple 概念&#xff1a;scala语言采用特殊的方式将无关的数据作为一个整体&#xff0c;组合在一起’ 1.元…

【Git】Git学习-13:Gitee和GitLab的使用

学习视频链接&#xff1a;【GeekHour】一小时Git教程_哔哩哔哩_bilibili​编辑https://www.bilibili.com/video/BV1HM411377j/?vd_source95dda35ac10d1ae6785cc7006f365780 流程 1. 创建仓库/已有仓库 2. 克隆到本地/在远程仓库关联 git clone 仓库地址 git remote add 仓库别…

小语言模型的潜力

想象一下这样一个世界&#xff1a;智能助手不在云端&#xff0c;而是在你的手机上&#xff0c;无缝了解你的需求并以闪电般的速度做出响应。这不是科幻小说&#xff0c;而是科幻小说。这是小语​​言模型 (SLM) 的希望&#xff0c;这是一个快速发展的领域&#xff0c;有可能改变…

聪明与诚实:社会信任的桥梁

在现代社会中&#xff0c;我们经常听到这样的评价&#xff1a;“某人真聪明。”然而&#xff0c;当我们深入思考时&#xff0c;会发现“聪明”这个词背后所承载的含义并不单一。聪明和狡诈往往被混淆&#xff0c;而诚实的价值却时常被忽视。在一个高度诚信的社会里&#xff0c;…

CentOS 自建gitlab仓库:安装相关工具

所需环境 Node 安装项目依赖、项目打包运行Nginx 前端项目部署&#xff08;正向代理、反向代理、负载均衡等&#xff09;Git 自动化部署时 拉取代码使用GitLab 代码仓库GitLab-Runner GitLab的CI/CD执行器 一、安装Node 检测是否已安装 常用node -v 命令检测。 如果已安装&a…

【挑战30天首通《谷粒商城》】-【第一天】03、简介-分布式基础概念

文章目录 课程介绍 ( 本章了解即可&#xff0c;可以略过)1、微服务简而言之: 2、集群&分布式&节点2.1、定义2.2、示例 3、远程调用4、负载均衡常见的负裁均衡算法: 5、服务注册/发现&注册中心6、配置中心7、服务熔断&服务降级7.1、服务熔断7.2、服务降级 8、AP…

NVIDIA: RULER新测量方法让大模型现形

1 引言 最近在人工智能系统工程和语言模型设计方面的进展已经实现了语言模型上下文长度的高效扩展。以前的工作通常采用合成任务,如密钥检索和大海捞针来评估长上下文语言模型(LMs)。然而,这些评估在不同工作中使用不一致,仅揭示了检索能力,无法衡量其他形式的长上下文理解。 …

(1day)致远M3 log 敏感信息泄露漏洞(Session)复现

前言 系统学习web漏洞挖掘以及项目实战也有一段时间了,发现在漏洞挖掘过程中难免会碰到一些历史漏洞,来帮助自己或是提高自己挖洞和及时发现漏洞效率,于是开始创建这个专栏,对第一时间发现的1day以及历史漏洞进行复现,来让自己更加熟悉漏洞类型以及历史漏洞,方便自己在后续的项…

为什么会查询不到DNS信息?怎么排查?

DNS&#xff08;域名系统&#xff09;是将域名转换为相应 IP 地址的关键系统。查询 DNS 信息具有重要作用&#xff0c;通过查询 DNS 信息&#xff0c;我们可以知道域名对应的 IP 地址&#xff0c;这是最主要的信息&#xff0c;使设备能与目标服务器进行通信&#xff1b;其次是域…

微信小程序03: 获取不限制的小程序二维码

全文目录,一步到位 1.前言简介1.1 专栏传送门1.1.1 上文小总结1.1.2 上文传送门 2. 获取不限制二维码操作2.1 准备工作2.1.1 请先复制00篇的统一封装代码2.1.2 修改配置文件中的参数 2.2 具体代码使用与注释如下2.2.1 业务代码如下2.2.2 代码解释(一)[无需复制]2.2.3 创建Base6…

每日一题5:Pandas-修改列

一、每日一题 一家公司决定增加员工的薪水。 编写一个解决方案&#xff0c;将每个员工的薪水乘以2来 修改 salary 列。 返回结果格式如下示例所示。 解答&#xff1a; import pandas as pddef modifySalaryColumn(employees: pd.DataFrame) -> pd.DataFrame:employees.loc[…

C语言之数据结构之栈和队列的运用

目录 1. 用队列实现栈1.1 思路讲解1.2 代码实现 2. 用栈实现队列1.1 思路讲解1.2 代码实现 总结 •͈ᴗ•͈ 个人主页&#xff1a;御翮 •͈ᴗ•͈ 个人专栏&#xff1a;C语言数据结构 •͈ᴗ•͈ 欢迎大家关注和订阅!!! 1. 用队列实现栈 题目描述&#xff1a; 请你仅使用两个…