设计模式——行为型模式——策略模式(含实际业务使用示例、可拷贝直接运行)

目录

策略模式

定义

组成和UML图

代码示例

实际业务场景下策略模式的使用

策略模式优缺点

使用场景

JDK中使用策略模式示例

参考文档

策略模式

定义

        策略模式定义了一系列算法,并将每个算法封装起来,使它们可以相互替换,且算法的变化不会影响使用算法的客户。

        策略模式属于对象行为模式,它通过对算法进行封装,把使用算法的责任和算法的实现分割开来,并委派给不同的对象对这些算法进行管理。

        考虑如下场景:去旅游选择出行方式时,可以骑自行车、坐汽车、坐火车、坐飞机,在不考虑使用策略模式的时候使用if elseif else语句进行判断,此时代码中存在大量条件判断语句,代码不便于增强,不符合开闭原则。将其修改为策略模式可以去掉代码中大量的if判断语句,同时增强代码的可扩展性。

组成和UML图

策略模式的组成

  • 抽象策略(Strategy)类:这是一个抽象角色,通常由一个接口或抽象类实现。此角色给出所有的具体策略类所需的接口。

  • 具体策略(Concrete Strategy)类:实现了抽象策略定义的接口,提供具体的算法实现或行为。

  • 环境(Context)类:持有一个策略类的引用,最终给客户端调用。

策略模式的UML图

代码示例

        促销活动:一家百货公司在定年度的促销活动。针对不同的节日(春节、中秋节、圣诞节)推出不同的促销活动,由促销员将促销活动展示给客户。类图如下:

        代码实现:以下示例代码仅仅是一个示例,便于理解策略模式,根据我的实习经验具体公司代码中策略模式不会这样使用的。【最近时间太紧,后期会梳理一下在实习中碰到的公司代码使用策略模式的例子,并更新以下代码示例(2024-05-08已更新,见下文)】

package com.scut.adapter.strategy;

// 定义抽象策略类
public interface Strategy {
    void show();
}

package com.scut.adapter.strategy;
// 策略实现类A
public class StrategyA implements Strategy{
    @Override
    public void show() {
        System.out.println("中秋节活动,赠送中秋大礼包...");
    }
}

package com.scut.adapter.strategy;

// 策略实现类B
public class StrategyB implements Strategy{
    @Override
    public void show() {
        System.out.println("充值一百,送二百...");
    }
}

package com.scut.adapter.strategy;

// 策略实现类C
public class StrategyC implements Strategy{
    @Override
    public void show() {
        System.out.println("圣诞节活动,赠送水果大礼盒一套...");
    }
}

package com.scut.adapter.strategy;

// 定义售卖员,策略上下文类
public class SaleMan {
    private Strategy strategy;

    public SaleMan(Strategy strategy) {
        this.strategy = strategy;
    }

    public Strategy getStrategy() {
        return strategy;
    }

    public void setStrategy(Strategy strategy) {
        this.strategy = strategy;
    }
    public void saleShow(){
        strategy.show();
    }
}

package com.scut.adapter.strategy;

// 定义客户端、测试类
public class Client {
    public static void main(String[] args) {
        SaleMan saleMan = new SaleMan(new StrategyA());
        saleMan.saleShow();
        System.out.println("====================");
        saleMan.setStrategy(new StrategyB());
        saleMan.saleShow();
        System.out.println("====================");
    }
}
实际业务场景下策略模式的使用

        前置说明:公司中策略模式的使用不会按照以上代码示例的方式使用,更多是结合@Autowire注解注入一系列接口实现类构成一个List集合,进而遍历List集合中的实现类选择具体方法。如下:

        模拟业务场景:电商业务通常需要和发票进行交互,发票简单状态包括:开票、改票、退票。将其抽象并结合策略模式进行设计可以有效减少代码中if else条件判断语句的数量并提高代码可读性。

代码结构如下:

具体代码如下:

package com.example.webmodule;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("/hello")
public class InvoiceController {

    @Autowired
    private InvoiceService invoiceService;

    @GetMapping("")
    public String testInvoice(int invoiceStatus){
        invoiceService.invoiceTest(invoiceStatus);
        return "OK";
    }
}



package com.example.webmodule;

public interface InvoiceHandle {
    // 实现类选择器
    boolean canHandle(int invoiceStatus);
    // 具体业务实现
    void invoiceProcess();
}


package com.example.webmodule;

import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.util.List;

@Service
@Slf4j
public class InvoiceService {
    @Autowired
    List<InvoiceHandle> invoiceHandleList;

    public void invoiceTest(int invoiceStatus){
        for (InvoiceHandle invoiceHandle : invoiceHandleList) {
            if(invoiceHandle.canHandle(invoiceStatus)){
                invoiceHandle.invoiceProcess();
            }
        }
    }
}



package com.example.webmodule;

public enum InvoiceStatusEnum {
    MAKE_INVOICE(1, "开票"),
    UPDATE_INVOICE(2, "改票"),
    REFUND_INVOICE(3,"退票");
    private int code;
    private String value;
    InvoiceStatusEnum(int code, String value) {
        this.code = code;
        this.value = value;
    }
    public int getCode(){
        return code;
    }
    public String getValue(){
        return value;
    }
    public void setCode(int code){
        this.code = code;
    }
    public void setValue(String value){
        this.value = value;
    }
}


package com.example.webmodule;

import org.springframework.stereotype.Service;

/**
 * 开票
 */
@Service
public class MakeInvoiceImpl implements InvoiceHandle{
    @Override
    public boolean canHandle(int invoiceStatus) {
        return InvoiceStatusEnum.MAKE_INVOICE.getCode() == invoiceStatus;
    }

    @Override
    public void invoiceProcess() {
        System.out.println("正在开票...");
    }
}


package com.example.webmodule;

import org.springframework.stereotype.Service;

/**
 * 退票
 */
@Service
public class RefundInvoiceImpl implements InvoiceHandle{
    @Override
    public boolean canHandle(int invoiceStatus) {
        return InvoiceStatusEnum.REFUND_INVOICE.getCode() == invoiceStatus;
    }

    @Override
    public void invoiceProcess() {
        System.out.println("正在退票...");
    }
}



package com.example.webmodule;

import org.springframework.stereotype.Service;

/**
 * 改票
 */
@Service
public class UpdateInvoiceImpl implements InvoiceHandle{
    @Override
    public boolean canHandle(int invoiceStatus) {
        return InvoiceStatusEnum.UPDATE_INVOICE.getCode() == invoiceStatus;
    }

    @Override
    public void invoiceProcess() {
        System.out.println("正在改票...");
    }
}

代码演示效果如下:

策略模式优缺点

优点:

  • 策略类之间可以自由切换

    由于策略类都实现同一个接口,所以使它们之间可以自由切换。

  • 易于扩展

    增加一个新的策略只需要添加一个具体的策略类即可,基本不需要改变原有的代码,符合“开闭原则“

  • 避免使用多重条件选择语句(if else),充分体现面向对象设计思想。

缺点:

  • 客户端必须知道所有的策略类,并自行决定使用哪一个策略类。

  • 策略模式将造成产生很多策略类,可以通过使用享元模式在一定程度上减少对象的数量。

使用场景
  • 一个系统需要动态地在几种算法中选择一种时,可将每个算法封装到策略类中。

  • 一个类定义了多种行为,并且这些行为在这个类的操作中以多个条件语句的形式出现,可将每个条件分支移入它们各自的策略类中以代替这些条件语句。

  • 系统中各算法彼此完全独立,且要求对客户隐藏具体算法的实现细节时。

  • 系统要求使用算法的客户不应该知道其操作的数据时,可使用策略模式来隐藏与算法相关的数据结构。

  • 多个类只区别在表现行为不同,可以使用策略模式,在运行时动态选择具体要执行的行为。

JDK中使用策略模式示例

        在Arrays类中的 sort() 方法中的Comparator使用的就是策略模式,具体源码可以自行查找以下。

参考文档

策略模式介绍及其具体使用场景_策略模式应用场景-CSDN博客

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

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

相关文章

【微信小程序开发】微信小程序、大前端之flex布局方式详细解析

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

液晶高抗干扰驱动LCD段码屏驱动芯片VK2C22抗干扰系列瓦斯表段码LCD液晶驱动芯片

VK2C22是一个点阵式存储映射的LCD驱动器&#xff0c;可支持最大176点&#xff08;44SEGx4COM&#xff09;的LCD屏。单片机可通过I2C接口配置显示参数和读写显示数据&#xff0c;也可通过指令进入省电模式。其高抗干扰&#xff0c;低功耗的特性适用于水电气表以及工控仪表类产品…

简单几步解决Windows 10播放视频提示安装HEVC扩展

相信有不少人都遇到过以下的问题&#xff0c;废话不多说&#xff0c;直接上干货&#xff01; 1.下载插件 免费地址链接: 点击下载 2.安装插件 如图所示&#xff0c;在下载的目录路径里&#xff0c; 1.按住键盘 SHIFT&#xff0c;点击鼠标右键&#xff0c;选择在此处打开Powe…

4WRPH系列比例阀外置放大器

控制4WRPH6或4WRPH10比例伺服阀放大器适用于驱动带非线性曲线的直动式比例伺服电磁阀&#xff0c;模拟量控制电器放大器模块式的放大器用于安装在机柜内35mm卡轨架上&#xff0c;输出级带电气反馈用于闭环控制。使能输入功能可控制放大器输出开或关&#xff0c;带斜坡时间发生器…

const成员函数、cout/cin和重载运算符<<、>>、

目录 一、为什么cout&#xff0c;cin可以自动识别类型&#xff1f; 二、留提取运算符重载&#xff08;<<&#xff09; 三、留插入运算符重载&#xff08;>>&#xff09; 四、对上述的总结&#xff1a; 五、const成员 成员函数原则&#xff1a; 六、默认成员函…

Object类

Object类 概念&#xff1a;Object类是所有类的父类&#xff0c;也就是说任何一个类在定义时候如果没有明确的指定继承一个父类的话&#xff0c;那么它就都默认继承Object类&#xff0c;因此Object类被称为所有类的父类&#xff0c;也叫做基类/超类。 常用方法 方法类型描述eq…

Python实战开发及案例分析(12)—— 模拟退火算法

模拟退火算法&#xff08;Simulated Annealing&#xff09;是一种概率搜索算法&#xff0c;源自于金属退火过程。在金属退火中&#xff0c;通过缓慢降低温度&#xff0c;金属内部的原子能够从高能态逐步达到较低能态。模拟退火算法利用类似的原理&#xff0c;通过随机搜索和概率…

Samtec连接器应用科普 | 连接智能工厂中的AI

【摘要/前言】 本文是系列的第一部分&#xff0c;我们将探讨人工智能在工业领域的作用。 人工智能&#xff08;AI&#xff09;的话题最近成为头条新闻&#xff0c;因为最新一代基于云的人工智能工具有望为机器的力量带来重大飞跃。在所有关于人工智能将如何影响我们的讨论中&…

Android内核之Binder消息处理:binder_transaction用法实例(七十三)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

overflow:hidden对解决外边距塌陷的个人理解

外边距塌陷&#xff1a; 子元素的上外边距大于父元素的上外边距&#xff0c;导致边距折叠&#xff0c;取两者之间最大值&#xff0c;即子元素外边距&#xff0c;导致父元素上外边距失效。 解决办法&#xff1a;在父元素样式添加overflow:hidden;或者border:1px solid black;(不…

Python数据分析实战

文章目录 第1关&#xff1a;读取MoMA数据集第2关&#xff1a;计算艺术家年龄第3关&#xff1a;把年龄换算成年代第4关&#xff1a;总结年代数据第5关&#xff1a;将变量插入字符串第6关&#xff1a;创建艺术家频率表第7关&#xff1a;创建显示艺术家信息的函数第8关&#xff1a…

Ubuntu下halcon软件的下载安装

由于工作需求&#xff0c;点云配准需要使用halcon进行实现&#xff0c;并且将该功能放入QT界面中 1.下载halcon 进入halcon官网进行下载 官网链接&#xff1a;https://www.mvtec.com/products/halcon/ 注意&#xff1a;要注册登陆之后才能进行下载 接着点击Downloads->H…

SOCKET编程(3):相关结构体与函数

相关结构体与函数 sockaddr、sockaddr_in结构体 sockaddr和sockaddr_in详解 struct sockaddr共16字节&#xff0c;协议族(family)占2字节&#xff0c;IP地址和端口号在sa_data字符数组中 /* Structure describing a generic socket address. */ struct sockaddr {__SOCKADDR…

Apache POI入门学习

Apache POI入门学习 官网地址 excel中使用到的类读取excel表格内容表格内容maven依赖方式一测试结果 方式二测试结果 向excel中写入数据方式一方式二方式三测试结果 从 Excel 工作表中的公式单元格读取数据测试结果 Excel 工作表中写入公式单元格从受密码保护的Excel中读取数据…

Apple 发布新款 iPad Pro 和 iPad Air:性能和设计的巨大飞跃

Apple 发布新款 iPad Pro 和 iPad Air&#xff1a;性能和设计的巨大飞跃 概述 苹果公司最近的“Let Loose”活动在科技界掀起了轩然大波&#xff0c;推出了最新的 iPad Pro 和 iPad Air 型号&#xff0c;在性能、设计和功能方面取得了前所未有的改进。在本文中&#xff0c;我…

【XR806开发板试用】使用FDCM操作Flash记录开机次数

一、寻找系统分配的自定义用户数据地址 &#xff08;1&#xff09;XR806的Flash布局 如图1所示&#xff0c;FLASH的布局有两种&#xff1a; 1、没有开启OTA模式&#xff1b;Image1PaddingSysinfo 2、开启OTA模式&#xff1b;Image1PaddingSysinfoOTA area Image2 Padding 如图…

智算中心“火”了?引领算力发展新潮流

去年大模型的空前发展&#xff0c;人工智能也终于迎来了属于自己的“文艺复兴”&#xff0c;众多的模型相继发布&#xff0c;继而催生了整个行业对于智能算力需求的激增。 市场需求与技术驱动仿佛现实世界的左右脚&#xff0c;催动着世界文明的齿轮向前滚动。在全球经济角逐日…

django中的cookie与session

获取cookie request.COOKIE.GET 使用cookie response.set-cookie views.py from django.http import HttpResponse from django.shortcuts import render# Create your views here. def cookie_test(request):r HttpResponse("hello world")r.set_cookie(lan, py…

AQ6360 横河 光谱分析仪精华帖,收藏保存

AQ6360是一款由日本横河&#xff08;YOKOGAWA&#xff09;生产的光谱分析仪&#xff0c;其主要技术参数包括波长范围、波长精度和波长线性度等。AQ6360的波长范围为1200~1650nm &#xff0c;具有较高的波长精度&#xff0c;在1520~1580nm范围内为0.02nm&#xff0c;在1580~1620…

Colab/PyTorch - 001 PyTorch Basics

Colab/PyTorch - 001 PyTorch Basics 1. 源由2. PyTorch库概览3. 处理过程2.1 数据加载与处理2.2 构建神经网络2.3 模型推断2.4 兼容性 3. 张量介绍3.1 构建张量3.2 访问张量元素3.3 张量元素类型3.4 张量转换&#xff08;NumPy Array&#xff09;3.5 张量运算3.6 CPU v/s GPU …