首页 > 其他分享 >设计模式之策略模式

设计模式之策略模式

时间:2022-12-06 21:35:56浏览次数:36  
标签:策略 模式 算法 具体 抽象 设计模式 类中

概述

在策略模式中,可以定义一些独立的类来封装不同的算法,每一个类封装一种具体的算法。在这里,每一个封装算法的类都可以称之为一种策略(Strategy),为了保证这些策略在使用时具有一致性,一般会提供一个抽象的策略类来做规则的定义,而每种算法则对应于一个具体策略类。

策略模式的主要目的是将算法的定义与使用分开,也就是将算法的行为和环境分开,将算法的定义放在专门的策略类中,每一个策略类封装了一种实现算法,使用算法的环境类针对抽象策略类进行编程,符合依赖倒转原则。在出现新的算法时,只需要增加一个新的实现了抽象策略类的具体策略类即可。

策略模式定义如下:定义一系列算法类,将每一个算法封装起来,并让它们可以相互替换。策略模式让算法独立于使用它的客户而变化,也称为政策模式(Policy)。策略模式是一种对象行为型模式。

策略模式结构并不复杂,但需要理解其中环境类 Context 的作用,其结构如图所示:

从图中可以看出,在策略模式结构图中包含以下 3 个角色:

  1. Context(环境类):环境类是使用算法的角色,它在解决某个问题(即实现某个方法)时可以采用多种策略。在环境类中维持一个对抽象策略类的引用实例,用于定义所采用的策略。
  2. Strategy(抽象策略类):它为所支持的算法声明了抽象方法,是所有策略类的父类,它可以是抽象类或具体类,也可以是接口。环境类通过抽象策略类中声明的方法在运行时调用具体策略类中实现的算法。
  3. ConcreteStrategy(具体策略类):它实现了在抽象策略类中声明的算法,在运行时,具体策略类将覆盖在环境类中定义的抽象策略类对象,使用一种具体的算法实现某个业务处理。

策略模式是一个比较容易理解和使用的设计模式。策略模式是对算法的封装,它把算法的责任和算法本身分割开,委派给不同的对象管理。策略模式通常把一个系列的算法封装到一系列具体策略类里面,作为抽象策略类的子类。在策略模式中,对环境类和抽象策略类的理解非常重要,环境类是需要使用算法的类。在一个系统中可以存在多个环境类,它们可能需要重用一些相同的算法。

在使用策略模式时,需要将算法从 Context 类中提取出来,首先应该创建一个抽象策略类,其典型代码如下:

class AbstractStrategy {
public:
	// 声明抽象算法
	virtual void algorithm() = 0;
};

然后再将封装每一种具体的算法的类作为该抽象策略类的子类,代码如下:

class ConcreteStrategyA : public AbstractStrategy {
public:
	// 算法的具体实现
	virtual void algorithm() override;
};

其它具体策略类与之类似,对于 Context 类而言,在它与抽象策略类之间建立一个关联类型,其典型代码如下:

class Context {
public:
	void setStrategy(AbstractStrategy* s) { strategy = s; }

	// 调用策略中的算法
	void algorithm() { strategy->algorithm(); }

private:
	AbstractStrategy* strategy;
};

在 Context 类中定义一个 AbstractStrategy 类型的对象 strategy,通过注入的方式在客户端传人一个具体策略对象,客户端代码片段如下:

...
Context context;
AbstractStrategy* strategy = new ConcreteStrategyA();
context.setStrategy(strategy);
context.algorithm();
...

在客户端代码中只需注入一个具体策略对象,可以将具体策略类类名存储在配置文件中,通过反射来动态创建具体策略对象,从而使得用户可以灵活地更换具体策略类,增加新的具体策略类也很方便。策略模式提供了一种可插入式(Pluggable)算法的实现方案。

总结

策略模式用于算法的自由切换和扩展,它是应用较为广泛的设计模式之一。策略模式对应于解决某一问题的一个算法族,允许用户从该算法族中任选一个算法来解决某一问题,同时可以方便地更换算法或者增加新的算法。只要涉及算法的封装、复用和切换都可以考虑使用策略模式。

优点

  1. 策略模式提供了对开闭原则的完美支持,用户可以在不修改原有系统的基础上选择算法或行为,也可以灵活地增加新的算法或行为。
  2. 策略模式提供了管理相关的算法族的办法。策略类的等级结构定义了一个算法或行为族,恰当使用继承可以把公共的代码移到抽象策略类中,从而避免重复的代码。
  3. 策略模式提供了一种可以替换继承关系的办法。如果不使用策略模式,那么使用算法的环境类就可能会有一些子类,每一个子类提供一种不同的算法。但是,这样一来算法的使用就和算法本身混在一起,不符合单一职责原则,决定使用哪一种算法的逻辑和该算法本身混合在一起,从而不可能再独立演化;而且使用继承无法实现算法或行为在程序运行时的动态切换。
  4. 使用策略模式可以避免多重条件选择语句。多重条件选择语句不易维护,它把采取哪一种算法或行为的逻辑与算法或行为本身的实现逻辑混合在一起,将它们全部硬编码(Hard Coding)在一个庞大的多重条件选择语句中,比直接继承环境类的办法还要原始和落后。
  5. 策略模式提供了一种算法的复用机制,由于将算法单独提取出来封装在策略类中,因此不同的环境类可以方便地复用这些策略类。

缺点

  1. 客户端必须知道所有的策略类,并自行决定使用哪一个策略类。这就意味着客户端必须理解这些算法的区别,以便适时选择恰当的算法。换言之,策略模式只适用于客户端知道所有的算法或行为的情况。
  2. 策略模式将造成系统产生很多具体策略类,任何细小的变化都将导致系统要增加一个新的具体策略类。
  3. 无法同时在客户端使用多个策略类,也就是说,在使用策略模式时,客户端每次只能使用一个策略类,不支持使用一个策略类完成部分功能后再使用另一个策略类来完成剩余功能的情况。

适用场景

  1. 一个系统需要动态地在几种算法中选择一种,那么可以将这些算法封装到一个个的具体算法类中,而这些具体算法类都是一个抽象算法类的子类。换言之,这些具体算法类均有统一的接口,根据里氏代换原则和面向对象的多态性,客户端可以选择使用任何一个具体算法类,并只需要维持一个数据类型是抽象算法类的对象。
  2. 一个对象有很多的行为,如果不用恰当的模式,这些行为就只好使用多重条件选择语句来实现。此时,使用策略模式,把这些行为转移到相应的具体策略类里面,就可以避免使用难以维护的多重条件选择语句。
  3. 不希望客户端知道复杂的、与算法相关的数据结构,在具体策略类中封装算法与相关的数据结构,可以提高算法的保密性与安全性。

标签:策略,模式,算法,具体,抽象,设计模式,类中
From: https://www.cnblogs.com/tuilk/p/16960611.html

相关文章

  • 五种常见的电子商务模式对比:B2B、B2C、C2B、C2C、O2O
    前言在接触和谈论到电商产品时,我们常常会听到一些专业术语,比如说B2B、B2C等,那么这些术语具体代表什么含义呢?通过收集和整理网络资料以及个人理解,展开说说:(1)B2B模式B2B(B......
  • 设计模式——23种设计模式
    Java设计模式设计模式:是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。面向对象设计原则都是为了高内聚低耦合原则。编程时基本都要遵守单一......
  • oracle19c pdb 模式下的dataguard 配置[1+1]
    标签(空格分隔):Oracle系列一:配置介绍使用Oracle19c的安装包安装两台主机:系统介绍:CentOS7.9x64cat/etc/hosts--------172.130.216.176[主库]1......
  • GEE推出收费模式,国产遥感云平台大有可为!
    谷歌地球引擎(GoogleEarthEngine,GEE)是一个用于对地理空间数据集进行科学分析和可视化的平台。科学家、研究人员和开发人员可以公开访问PB级的卫星图像和地理空间数据,用于全......
  • Java的两大、三类代理模式
    简述代理,是一种设计模式,主要作用是为其他对象提供一种代理,以控制对这个对象的访问。在某些情况下,一个对象不想或者不能直接引用另一个对象,而代理对象可以在客户端和目......
  • 05.Nodejs_web开发模式
    Web开发目录目录Web开发Web开发模式服务端渲染的传统Web开发模式前后端分离的Web开发模式Session认证机制在项目中使用Session向session中存入数据:从session中获取数......
  • Redis主从复制,哨兵模式和集群模式
    一、主从复制1、主从复制-哨兵-集群主从复制:主从复制是高可用Redis的基础,哨兵和集群都是在主从复制基础上实现高可用的。主从复制主要实现了数据的多机备份,以及对于读操......
  • 设计模式结构型[下]
    外观模式概念外观模式,通过为多个复杂的子系统提供一个一致的接口,使得这些子系统更容易被访问。外部不用关心内部系统的细节,只是使用这个统一的接口就足够了类图代码classF......
  • 网络爬虫优化几种常见策略
    什么是网络爬虫和反爬虫:网络爬虫:使用一些技术手段,大量获取网站数据的一种方式。反爬虫:使用一些技术手段,阻止网络爬虫获取次网站数据的一种方式。常见的反爬虫机制有哪些:通过......
  • 20.备忘录模式
    [实验任务一]:多次撤销改进课堂上的“用户信息操作撤销”实例,使得系统可以实现多次撤销(可以使用HashMap、ArrayList等集合数据结构实现)。代码JavaCaretaker.javapackag......